* @license http://opensource.org/licenses/osl-3.0.php Open Software License (OSL 3.0) * @copyright PayPal */ if (!defined('_PS_VERSION_')) { exit; } require_once _PS_MODULE_DIR_ . 'paypal/config_prod.php'; include_once _PS_MODULE_DIR_ . 'paypal/vendor/autoload.php'; use PaypalAddons\classes\AbstractMethodPaypal; use PaypalAddons\classes\ACDC\AcdcFunctionality; use PaypalAddons\classes\ACDC\AcdcPaymentMethod; use PaypalAddons\classes\APM\ApmCollection; use PaypalAddons\classes\APM\ApmFunctionality; use PaypalAddons\classes\Constants\APM; use PaypalAddons\classes\Constants\WebHookConf; use PaypalAddons\classes\InstallmentBanner\BannerManager; use PaypalAddons\classes\InstallmentBanner\BNPL\BnplAvailabilityManager; use PaypalAddons\classes\InstallmentBanner\BNPL\BNPLCart; use PaypalAddons\classes\InstallmentBanner\BNPL\BNPLDummy; use PaypalAddons\classes\InstallmentBanner\BNPL\BNPLOption; use PaypalAddons\classes\InstallmentBanner\BNPL\BNPLPaymentStep; use PaypalAddons\classes\InstallmentBanner\BNPL\BNPLProduct; use PaypalAddons\classes\InstallmentBanner\BNPL\BNPLSignup; use PaypalAddons\classes\InstallmentBanner\ConfigurationMap; use PaypalAddons\classes\InstallmentBanner\ConfigurationMap as InstallmentConfiguration; use PaypalAddons\classes\PaypalPaymentMode; use PaypalAddons\classes\PUI\FraudNetForm; use PaypalAddons\classes\PUI\FraudSessionId; use PaypalAddons\classes\PUI\PuiFunctionality; use PaypalAddons\classes\SEPA\SepaButton; use PaypalAddons\classes\Shortcut\ShortcutConfiguration; use PaypalAddons\classes\Shortcut\ShortcutPaymentStep; use PaypalAddons\classes\Shortcut\ShortcutSignup; use PaypalAddons\classes\Venmo\VenmoButton; use PaypalAddons\classes\Venmo\VenmoFunctionality; use PaypalAddons\classes\Webhook\WebhookOption; use PaypalAddons\classes\Widget\DummyWidget; use PaypalAddons\classes\Widget\InstallmentWidget; use PaypalAddons\classes\Widget\ShortcutWidget; use PaypalAddons\services\PaymentRefundAmount; use PaypalAddons\services\PaypalContext; use PaypalAddons\services\ServicePaypalOrder; use PaypalAddons\services\StatusMapping; use PaypalAddons\services\WebhookService; use PaypalPPBTlib\Extensions\AbstractModuleExtension; use PaypalPPBTlib\Extensions\ProcessLogger\ProcessLoggerExtension; use PaypalPPBTlib\Extensions\ProcessLogger\ProcessLoggerHandler; use PaypalPPBTlib\Install\ModuleInstaller; use PrestaShop\PrestaShop\Core\Module\WidgetInterface; use PrestaShop\PrestaShop\Core\Payment\PaymentOption; define('BT_CARD_PAYMENT', 'card-braintree'); define('BT_PAYPAL_PAYMENT', 'paypal-braintree'); define('PAYPAL_PAYMENT_CUSTOMER_CURRENCY', -1); // Method Alias : // EC = express checkout // ECS = express checkout sortcut // BT = Braintree // PPP = PayPal Plus class PayPal extends \PaymentModule implements WidgetInterface { const PAYPAL_PARTNER_CLIENT_ID_LIVE = 'ATgR8ZE5M_Jd7F_XMMQDqMfFFgr7hJHFw8yKfklWU4TwzReENgydr5I042YfS1nRTDey7C1NbuFfKo_o'; const PAYPAL_PARTNER_ID_LIVE = 'B3PVCXSW2J8JN'; const PAYPAL_PARTNER_CLIENT_ID_SANDBOX = 'AVJ8YvTxw5Clf5CyJXIX6mnSSNgpzFFRaZh0KekLIMVe2vlkrWDMgaOTbvNds1U2bXVcjX4JGaP_jDM1'; const PAYPAL_PARTNER_ID_SANDBOX = 'J7Q7R6V9MQZUG'; const NEED_INSTALL_MODELS = 'PAYPAL_NEED_INSTALL_MODELS'; public static $dev = true; public $express_checkout; public $message; public $amount_paid_paypal; public $module_link; public $errors; public $currencyMB = ['USD', 'MXN', 'EUR', 'BRL']; public $paypal_method; public $countriesApiCartUnavailable = ['FR', 'ES', 'IT', 'GB', 'PL', 'BE', 'NL', 'LU', 'US', 'GR', 'DK', 'CZ', 'PT', 'FI', 'SE', 'NO', 'SK', 'CY', 'EE', 'LV', 'LT', 'MT', 'SI']; /** @var array matrix of state iso codes between paypal and prestashop */ public static $state_iso_code_matrix = [ 'MX' => [ 'AGS' => 'AGS', 'BCN' => 'BC', 'BCS' => 'BCS', 'CAM' => 'CAMP', 'CHP' => 'CHIS', 'CHH' => 'CHIH', 'COA' => 'COAH', 'COL' => 'COL', 'DIF' => 'DF', 'DUR' => 'DGO', 'GUA' => 'GTO', 'GRO' => 'GRO', 'HID' => 'HGO', 'JAL' => 'JAL', 'MEX' => 'MEX', 'MIC' => 'MICH', 'MOR' => 'MOR', 'NAY' => 'NAY', 'NLE' => 'NL', 'OAX' => 'OAX', 'PUE' => 'PUE', 'QUE' => 'QRO', 'ROO' => 'Q ROO', 'SLP' => 'SLP', 'SIN' => 'SIN', 'SON' => 'SON', 'TAB' => 'TAB', 'TAM' => 'TAMPS', 'TLA' => 'TLAX', 'VER' => 'VER', 'YUC' => 'YUC', 'ZAC' => 'ZAC', ], 'JP' => [ 'Aichi' => 'Aichi-KEN', 'Akita' => 'Akita-KEN', 'Aomori' => 'Aomori-KEN', 'Chiba' => 'Chiba-KEN', 'Ehime' => 'Ehime-KEN', 'Fukui' => 'Fukui-KEN', 'Fukuoka' => 'Fukuoka-KEN', 'Fukushima' => 'Fukushima-KEN', 'Gifu' => 'Gifu-KEN', 'Gunma' => 'Gunma-KEN', 'Hiroshima' => 'Hiroshima-KEN', 'Hokkaido' => 'Hokkaido-KEN', 'Hyogo' => 'Hyogo-KEN', 'Ibaraki' => 'Ibaraki-KEN', 'Ishikawa' => 'Ishikawa-KEN', 'Iwate' => 'Iwate-KEN', 'Kagawa' => 'Kagawa-KEN', 'Kagoshima' => 'Kagoshima-KEN', 'Kanagawa' => 'Kanagawa-KEN', 'Kochi' => 'Kochi-KEN', 'Kumamoto' => 'Kumamoto-KEN', 'Kyoto' => 'Kyoto-KEN', 'Mie' => 'Mie-KEN', 'Miyagi' => 'Miyagi-KEN', 'Miyazaki' => 'Miyazaki-KEN', 'Nagano' => 'Nagano-KEN', 'Nagasaki' => 'Nagasaki-KEN', 'Nara' => 'Nara-KEN', 'Niigata' => 'Niigata-KEN', 'Oita' => 'Oita-KEN', 'Okayama' => 'Okayama-KEN', 'Okinawa' => 'Okinawa-KEN', 'Osaka' => 'Osaka-KEN', 'Saga' => 'Saga-KEN', 'Saitama' => 'Saitama-KEN', 'Shiga' => 'Shiga-KEN', 'Shimane' => 'Shimane-KEN', 'Shizuoka' => 'Shizuoka-KEN', 'Tochigi' => 'Tochigi-KEN', 'Tokushima' => 'Tokushima-KEN', 'Tokyo' => 'Tokyo-KEN', 'Tottori' => 'Tottori-KEN', 'Toyama' => 'Toyama-KEN', 'Wakayama' => 'Wakayama-KEN', 'Yamagata' => 'Yamagata-KEN', 'Yamaguchi' => 'Yamaguchi-KEN', 'Yamanashi' => 'Yamanashi-KEN', ], ]; /** * List of objectModel used in this Module * * @var array */ public $objectModels = [ 'PaypalCapture', 'PaypalOrder', 'PaypalVaulting', 'PaypalIpn', 'PaypalWebhook', ]; /** * List of ppbtlib extentions */ public $extensions = [ ProcessLoggerExtension::class, ]; /** * List of hooks used in this Module */ public $hooks = [ 'paymentOptions', 'displayOrderConfirmation', 'displayAdminOrder', 'actionOrderStatusPostUpdate', 'actionOrderStatusUpdate', 'displayHeader', 'displayFooterProduct', 'actionBeforeCartUpdateQty', 'displayReassurance', 'displayInvoiceLegalFreeText', 'displayShoppingCartFooter', 'actionOrderSlipAdd', 'displayAdminOrderTabOrder', 'displayAdminOrderContentOrder', 'displayAdminCartsView', 'displayAdminOrderTop', 'displayAdminOrderTabLink', 'displayAdminOrderTabContent', 'displayOrderPreview', 'displayNavFullWidth', 'actionLocalizationPageSave', 'actionAdminOrdersTrackingNumberUpdate', ShortcutConfiguration::HOOK_REASSURANCE, ShortcutConfiguration::HOOK_AFTER_PRODUCT_ADDITIONAL_INFO, ShortcutConfiguration::HOOK_AFTER_PRODUCT_THUMBS, ShortcutConfiguration::HOOK_EXPRESS_CHECKOUT, ShortcutConfiguration::HOOK_FOOTER_PRODUCT, ShortcutConfiguration::HOOK_PRODUCT_ACTIONS, ShortcutConfiguration::HOOK_SHOPPING_CART_FOOTER, ShortcutConfiguration::HOOK_PERSONAL_INFORMATION_TOP, ]; /** * @var array */ public $moduleConfigs = []; /** * List of admin tabs used in this Module */ public $moduleAdminControllers = [ [ 'name' => [ 'en' => 'PayPal Official', 'fr' => 'PayPal Officiel', ], 'class_name' => 'AdminParentPaypalConfiguration', 'parent_class_name' => 'SELL', 'visible' => false, 'icon' => 'payment', ], [ 'name' => [ 'en' => 'Configuration', 'fr' => 'Configuration', ], 'class_name' => 'AdminPaypalConfiguration', 'parent_class_name' => 'AdminParentPaypalConfiguration', 'visible' => false, ], [ 'name' => [ 'en' => 'Setup', 'fr' => 'Paramètres', 'pt' => 'Definições', 'pl' => 'Ustawienia', 'nl' => 'Instellingen', 'it' => 'Impostazioni', 'es' => 'Configuración', 'de' => 'Einstellungen', 'mx' => 'Configuración', 'br' => 'Definições', ], 'class_name' => 'AdminPayPalSetup', 'parent_class_name' => 'AdminPayPalConfiguration', 'visible' => true, ], [ 'name' => [ 'en' => 'Experience', 'fr' => 'Expérience', 'de' => 'User Experience', 'pt' => 'Experiência', 'pl' => 'Doświadczenie', 'nl' => 'Ervaring', 'it' => 'Percorso Cliente', 'es' => 'Experiencia', 'mx' => 'Experiencia', 'br' => 'Experiência', ], 'class_name' => 'AdminPayPalCustomizeCheckout', 'parent_class_name' => 'AdminPayPalConfiguration', 'visible' => true, ], [ 'name' => [ 'en' => 'Pay in X times', 'fr' => 'Paiement en X fois', 'de' => 'Pay in X times', ], 'class_name' => 'AdminPayPalInstallment', 'parent_class_name' => 'AdminPayPalConfiguration', 'visible' => true, ], [ 'name' => [ 'en' => 'Help', 'fr' => 'Aide', 'pt' => 'Ajuda', 'pl' => 'Pomoc', 'nl' => 'Hulp', 'it' => 'Aiuto', 'es' => 'Ayuda', 'de' => 'Hilfe', 'mx' => 'Ayuda', 'br' => 'Ajuda', ], 'class_name' => 'AdminPayPalHelp', 'parent_class_name' => 'AdminPayPalConfiguration', 'visible' => true, ], [ 'name' => [ 'en' => 'Logs', 'fr' => 'Logs', 'de' => 'Logs', 'pt' => 'Logs', 'pl' => 'Dzienniki', 'nl' => 'Logs', 'it' => 'Logs', 'es' => 'Logs', 'mx' => 'Logs', 'br' => 'Logs', ], 'class_name' => 'AdminPayPalLogs', 'parent_class_name' => 'AdminPayPalConfiguration', 'visible' => true, ], [ 'name' => [ 'en' => 'Get Credentials', ], 'class_name' => 'AdminPaypalGetCredentials', 'parent_class_name' => 'AdminParentPaypalConfiguration', 'visible' => false, ], [ 'name' => [ 'en' => 'PUI listener', ], 'class_name' => 'AdminPayPalPUIListener', 'parent_class_name' => 'AdminParentPaypalConfiguration', 'visible' => false, ], ]; public function __construct() { $this->name = 'paypal'; $this->tab = 'payments_gateways'; $this->version = '5.7.2'; $this->author = '202 ecommerce'; $this->module_key = '336225a5988ad434b782f2d868d7bfcd'; $this->ps_versions_compliancy = ['min' => '1.7', 'max' => _PS_VERSION_]; $this->controllers = ['payment', 'validation']; $this->bootstrap = true; $this->currencies = true; $this->currencies_mode = 'radio'; parent::__construct(); require_once realpath(dirname(__FILE__) . '/smarty/plugins') . '/modifier.paypalreplace.php'; $this->displayName = $this->l('PayPal'); $this->description = $this->l('Allow your customers to pay with PayPal - the safest, quickest and easiest way to pay online.'); $this->confirmUninstall = $this->l('Are you sure you want to delete your details?'); $this->express_checkout = $this->l('PayPal Express Checkout '); $this->errors = ''; $countryDefault = new \Country((int) \Configuration::get('PS_COUNTRY_DEFAULT'), $this->context->language->id); switch ($countryDefault->iso_code) { case 'DE': $this->paypal_method = 'PPP'; break; case 'BR': $this->paypal_method = 'MB'; break; case 'MX': $this->paypal_method = 'MB'; break; default: $this->paypal_method = 'EC'; } $this->moduleConfigs = [ 'PAYPAL_MERCHANT_ID_SANDBOX' => '', 'PAYPAL_MERCHANT_ID_LIVE' => '', 'PAYPAL_USERNAME_SANDBOX' => '', 'PAYPAL_PSWD_SANDBOX' => '', 'PAYPAL_SIGNATURE_SANDBOX' => '', 'PAYPAL_SANDBOX_ACCESS' => 0, 'PAYPAL_USERNAME_LIVE' => '', 'PAYPAL_PSWD_LIVE' => '', 'PAYPAL_SIGNATURE_LIVE' => '', 'PAYPAL_LIVE_ACCESS' => 0, 'PAYPAL_SANDBOX' => 0, 'PAYPAL_API_INTENT' => 'sale', 'PAYPAL_API_ADVANTAGES' => 1, 'PAYPAL_API_CARD' => 1, 'PAYPAL_METHOD' => '', 'PAYPAL_EXPRESS_CHECKOUT_SHORTCUT' => 0, 'PAYPAL_EXPRESS_CHECKOUT_SHORTCUT_CART' => 1, 'PAYPAL_CRON_TIME' => '', 'PAYPAL_BY_BRAINTREE' => 0, 'PAYPAL_EXPRESS_CHECKOUT_IN_CONTEXT' => 1, 'PAYPAL_VAULTING' => 0, 'PAYPAL_REQUIREMENTS' => 0, 'PAYPAL_MB_EC_ENABLED' => 1, 'PAYPAL_CUSTOMIZE_ORDER_STATUS' => 0, 'PAYPAL_OS_REFUNDED' => (int) Configuration::get('PS_OS_REFUND'), 'PAYPAL_OS_CANCELED' => (int) Configuration::get('PS_OS_CANCELED'), 'PAYPAL_OS_ACCEPTED' => (int) Configuration::get('PS_OS_PAYMENT'), 'PAYPAL_OS_CAPTURE_CANCELED' => (int) Configuration::get('PS_OS_CANCELED'), 'PAYPAL_OS_ACCEPTED_TWO' => (int) Configuration::get('PS_OS_PAYMENT'), 'PAYPAL_OS_WAITING_VALIDATION' => (int) Configuration::get('PAYPAL_OS_WAITING'), 'PAYPAL_OS_PROCESSING' => (int) Configuration::get('PAYPAL_OS_WAITING'), 'PAYPAL_OS_VALIDATION_ERROR' => (int) Configuration::get('PS_OS_CANCELED'), 'PAYPAL_OS_REFUNDED_PAYPAL' => (int) Configuration::get('PS_OS_REFUND'), \PaypalAddons\classes\InstallmentBanner\ConfigurationMap::ENABLE_BNPL => 1, \PaypalAddons\classes\InstallmentBanner\ConfigurationMap::BNPL_CART_PAGE => 1, \PaypalAddons\classes\InstallmentBanner\ConfigurationMap::BNPL_PAYMENT_STEP_PAGE => 1, 'PAYPAL_NOT_SHOW_PS_CHECKOUT' => json_encode([$this->version, 0]), WebHookConf::ENABLE => 1, ]; if (version_compare(_PS_VERSION_, '1.7.6', '<')) { $this->moduleConfigs[ShortcutConfiguration::PRODUCT_PAGE_HOOK] = ShortcutConfiguration::HOOK_REASSURANCE; } else { $this->moduleConfigs[ShortcutConfiguration::PRODUCT_PAGE_HOOK] = ShortcutConfiguration::HOOK_PRODUCT_ACTIONS; } $this->moduleConfigs[ShortcutConfiguration::CART_PAGE_HOOK] = ShortcutConfiguration::HOOK_EXPRESS_CHECKOUT; } public function install() { $installer = new ModuleInstaller($this); $isPhpVersionCompliant = false; try { $isPhpVersionCompliant = $installer->checkPhpVersion(); } catch (\Exception $e) { $this->_errors[] = Tools::displayError($e->getMessage()); } if (($isPhpVersionCompliant && parent::install() && $installer->install()) == false) { return false; } // Registration order status if (!$this->installOrderState()) { return false; } $this->registerHooks(); $this->moduleConfigs['PAYPAL_OS_WAITING_VALIDATION'] = (int) Configuration::get('PAYPAL_OS_WAITING'); $this->moduleConfigs['PAYPAL_OS_PROCESSING'] = (int) Configuration::get('PAYPAL_OS_WAITING'); $shops = Shop::getShops(); foreach ($this->moduleConfigs as $key => $value) { if (Shop::isFeatureActive()) { foreach ($shops as $shop) { if (!Configuration::updateValue($key, $value, false, null, (int) $shop['id_shop'])) { return false; } } } else { if (!Configuration::updateValue($key, $value)) { return false; } } } if (Shop::isFeatureActive()) { $shops = Shop::getShops(); foreach ($shops as $shop) { Configuration::updateValue('PAYPAL_CRON_TIME', date('Y-m-d H:m:s'), false, null, (int) $shop['id_shop']); Configuration::updateValue('PAYPAL_PREVIOUS_VERSION', $this->version, false, null, (int) $shop['id_shop']); } } else { Configuration::updateValue('PAYPAL_CRON_TIME', date('Y-m-d H:m:s')); Configuration::updateValue('PAYPAL_PREVIOUS_VERSION', $this->version); } $this->hookActionLocalizationPageSave([]); return true; } /** * Set default currency restriction to "customer currency" * * @return bool */ public function updateRadioCurrencyRestrictionsForModule() { $shops = Shop::getShops(true, null, true); foreach ($shops as $s) { if (!Db::getInstance()->execute('UPDATE `' . _DB_PREFIX_ . 'module_currency` SET `id_currency` = -1 WHERE `id_shop` = "' . (int) $s . '" AND `id_module` = ' . (int) $this->id)) { return false; } } return true; } /** * Create order state * * @return bool */ public function installOrderState() { if (!Configuration::get('PAYPAL_OS_WAITING') || !Validate::isLoadedObject(new OrderState(Configuration::get('PAYPAL_OS_WAITING')))) { $order_state = new OrderState(); $order_state->name = []; foreach (Language::getLanguages() as $language) { if (Tools::strtolower($language['iso_code']) == 'fr') { $order_state->name[$language['id_lang']] = 'En attente de paiement PayPal'; } else { $order_state->name[$language['id_lang']] = 'Awaiting for PayPal payment'; } } $order_state->send_email = false; $order_state->color = '#4169E1'; $order_state->hidden = false; $order_state->delivery = false; $order_state->logable = false; $order_state->invoice = false; $order_state->module_name = $this->name; if ($order_state->add()) { $source = _PS_MODULE_DIR_ . 'paypal/views/img/os_paypal.png'; $destination = _PS_ROOT_DIR_ . '/img/os/' . (int) $order_state->id . '.gif'; copy($source, $destination); } if (Shop::isFeatureActive()) { $shops = Shop::getShops(); foreach ($shops as $shop) { Configuration::updateValue('PAYPAL_OS_WAITING', (int) $order_state->id, false, null, (int) $shop['id_shop']); } } else { Configuration::updateValue('PAYPAL_OS_WAITING', (int) $order_state->id); } } return true; } public function uninstall() { $installer = new ModuleInstaller($this); foreach ($this->moduleConfigs as $key => $value) { if (!Configuration::deleteByName($key)) { return false; } } if (parent::uninstall() == false) { return false; } if ($installer->uninstallModuleAdminControllers() == false) { return false; } return true; } /** * Delete order states * * @return bool */ public function uninstallOrderStates() { /* @var $orderState OrderState */ $result = true; $collection = new PrestaShopCollection('OrderState'); $collection->where('module_name', '=', $this->name); $orderStates = $collection->getResults(); if ($orderStates == false) { return $result; } foreach ($orderStates as $orderState) { $result &= $orderState->delete(); } return $result; } public function getUrl() { if (Configuration::get('PAYPAL_SANDBOX')) { return 'https://www.sandbox.paypal.com/'; } else { return 'https://www.paypal.com/'; } } public function hookDisplayShoppingCartFooter() { if ($this->context->controller instanceof CartController === false) { return ''; } return $this->displayShortcutButton([ 'sourcePage' => ShortcutConfiguration::SOURCE_PAGE_CART, 'hook' => ShortcutConfiguration::HOOK_SHOPPING_CART_FOOTER, ]); } public function hookDisplayProductActions($params) { $content = ''; $bannerManager = $this->getBannerManager(); if ($bannerManager->isBannerAvailable()) { $content .= $bannerManager->renderForProductPage(); } $content .= $this->displayShortcutButton([ 'sourcePage' => ShortcutConfiguration::SOURCE_PAGE_PRODUCT, 'hook' => ShortcutConfiguration::HOOK_PRODUCT_ACTIONS, ]); $content .= $this->renderBnpl(['sourcePage' => ShortcutConfiguration::SOURCE_PAGE_PRODUCT]); return $content; } public function hookDisplayAfterProductThumbs($params) { if ((int) Configuration::get(ShortcutConfiguration::DISPLAY_MODE_PRODUCT) !== ShortcutConfiguration::DISPLAY_MODE_TYPE_HOOK) { return ''; } return $this->displayShortcutButton([ 'sourcePage' => ShortcutConfiguration::SOURCE_PAGE_PRODUCT, 'hook' => ShortcutConfiguration::HOOK_AFTER_PRODUCT_THUMBS, ]); } public function hookDisplayProductAdditionalInfo($params) { return $this->displayShortcutButton([ 'sourcePage' => ShortcutConfiguration::SOURCE_PAGE_PRODUCT, 'hook' => ShortcutConfiguration::HOOK_AFTER_PRODUCT_ADDITIONAL_INFO, ]); } public function hookDisplayFooterProduct($params) { return $this->displayShortcutButton([ 'sourcePage' => ShortcutConfiguration::SOURCE_PAGE_PRODUCT, 'hook' => ShortcutConfiguration::HOOK_FOOTER_PRODUCT, ]); } public function hookDisplayExpressCheckout($params) { $returnContent = $this->displayShortcutButton([ 'sourcePage' => ShortcutConfiguration::SOURCE_PAGE_CART, 'hook' => ShortcutConfiguration::HOOK_EXPRESS_CHECKOUT, ]); $returnContent .= $this->renderBnpl(['sourcePage' => ShortcutConfiguration::SOURCE_PAGE_CART]); $bannerManager = $this->getBannerManager(); if ($bannerManager->isBannerAvailable()) { $returnContent .= $bannerManager->renderForCartPage(); } return $returnContent; } public function hookDisplayPersonalInformationTop($params) { if ($this->context->customer->isLogged() || $this->context->customer->is_guest) { return ''; } $content = $this->renderBnpl(['sourcePage' => ShortcutConfiguration::SOURCE_PAGE_SIGNUP]); $content .= $this->displayShortcutButton([ 'sourcePage' => ShortcutConfiguration::SOURCE_PAGE_SIGNUP, 'hook' => ShortcutConfiguration::HOOK_PERSONAL_INFORMATION_TOP, ]); return $content; } public function hookDisplayNavFullWidth() { if ($this->context->controller instanceof IndexController || $this->context->controller instanceof CategoryController) { $bannerManager = $this->getBannerManager(); if ($bannerManager->isBannerAvailable()) { return $bannerManager->renderForHomePage(); } } } public function getContent() { return Tools::redirectAdmin($this->context->link->getAdminLink('AdminPayPalSetup')); } protected function initVenmoFunctionality() { return new VenmoFunctionality(); } /** * @param $params * * @return array * * @throws Exception * @throws SmartyException */ public function hookPaymentOptions($params) { if (Module::isEnabled('braintreeofficial') && (int) Configuration::get('BRAINTREEOFFICIAL_ACTIVATE_PAYPAL')) { return []; } $isoCountryDefault = Country::getIsoById(Configuration::get('PS_COUNTRY_DEFAULT')); $payments_options = []; $method = AbstractMethodPaypal::load(); $bnplAvailabilityManager = $this->getBnplAvailabilityManager(); $bnplOption = $this->getBnplOption(); $venmoFunctionality = $this->initVenmoFunctionality(); switch ($this->paypal_method) { case 'EC': if ($method->isConfigured()) { $paymentOptionsEc = $this->renderEcPaymentOptions($params); $payments_options = array_merge($payments_options, $paymentOptionsEc); if (Configuration::get('PAYPAL_API_CARD') && (in_array($isoCountryDefault, $this->countriesApiCartUnavailable) == false)) { $payment_option = new PaymentOption(); $action_text = $this->l('Pay with debit or credit card'); $payment_option->setLogo(Media::getMediaPath(_PS_MODULE_DIR_ . $this->name . '/views/img/logo_card.png')); $payment_option->setCallToActionText($action_text); $payment_option->setModuleName($this->name); $payment_option->setAction($this->context->link->getModuleLink($this->name, 'ecInit', ['credit_card' => '1'], true)); $payment_option->setAdditionalInformation($this->context->smarty->fetch('module:paypal/views/templates/front/payment_infos_card.tpl')); $payments_options[] = $payment_option; } } break; case 'PPP': if ($method->isConfigured()) { if ((Configuration::get('PAYPAL_EXPRESS_CHECKOUT_SHORTCUT') || Configuration::get('PAYPAL_EXPRESS_CHECKOUT_SHORTCUT_CART')) && isset($this->context->cookie->paypal_pSc)) { $payment_option = new PaymentOption(); $action_text = $this->l('Pay with paypal plus shortcut'); $payment_option->setCallToActionText($action_text); $payment_option->setModuleName('paypal_plus_schortcut'); $payment_option->setAction($this->context->link->getModuleLink($this->name, 'pppValidation', ['short_cut' => '1', 'token' => $this->context->cookie->paypal_pSc], true)); $payments_options[] = $payment_option; } $payments_options[] = $this->buildPaypalWallet($params); } break; case 'MB': if (in_array($this->context->currency->iso_code, $this->currencyMB)) { if ((int) Configuration::get('PAYPAL_MB_EC_ENABLED')) { $methodEC = AbstractMethodPaypal::load('EC'); if ($methodEC->isConfigured()) { $paymentOptionsEc = $this->renderEcPaymentOptions($params); $payments_options = array_merge($payments_options, $paymentOptionsEc); } } if ($method->isConfigured() && (int) Configuration::get('PAYPAL_API_CARD') && (in_array($isoCountryDefault, $this->countriesApiCartUnavailable) == false)) { $payment_option = new PaymentOption(); $action_text = $this->l('Pay with credit or debit card'); $payment_option->setCallToActionText($action_text); $payment_option->setModuleName('paypal_plus_mb'); try { $this->context->smarty->assign('path', $this->_path); $payment_option->setAdditionalInformation($this->context->smarty->fetch('module:paypal/views/templates/front/payment_mb.tpl')); } catch (Exception $e) { return; } $payments_options[] = $payment_option; } } break; } if ($method->isConfigured()) { if ($bnplOption->isEnable() && $bnplOption->displayOnPaymentStep()) { if ($bnplAvailabilityManager->isEligibleCountryConfiguration() && $bnplAvailabilityManager->isEligibleContext()) { $payments_options[] = $this->buildBnplPaymentOption($params); } } if ($venmoFunctionality->isAvailable() && $venmoFunctionality->isEnabled() && $venmoFunctionality->isEligibleContext($this->context)) { $payments_options[] = $this->buildVenmoPaymentOption($params); } if ($this->initAcdcFunctionality()->isAvailable() && $this->initAcdcFunctionality()->isEnabled()) { $payments_options[] = $this->buildAcdcPaymentOption($params); } if ($this->initApmFunctionality()->isEnabled() && $this->initApmFunctionality()->isAvailable()) { $payments_options = array_merge($payments_options, $this->buildApmPaymentOptions($params)); } if ($this->paypal_method == 'PPP') { $payments_options[] = $this->renderSepaOption($params); if ($this->getWebhookOption()->isAvailable() && $this->getWebhookOption()->isEnable()) { if ($this->initPuiFunctionality()->isAvailable(false) && $this->initPuiFunctionality()->isEligibleContext($this->context)) { $payments_options[] = $this->renderPuiOption($params); } } } } if ($method->isSandbox() && false === empty($payments_options)) { foreach ($payments_options as $paymentOption) { if ($paymentOption instanceof PaymentOption) { $additionalInformantion = $this->displayAlert( $this->l('Sandbox mode: all transactions will be fictitious.'), 'info', false ); $additionalInformantion .= $paymentOption->getAdditionalInformation(); $paymentOption->setAdditionalInformation($additionalInformantion); } } } return $payments_options; } protected function buildVenmoPaymentOption($params = []) { $paymentOption = new PaymentOption(); $action_text = $this->l('Venmo'); $paymentOption->setCallToActionText($action_text); $paymentOption->setAction( sprintf( 'javascript:alert(\'%s\');', $this->l('Should use the button "Venmo"') // todo: specify message ) ); $paymentOption->setModuleName('paypal_venmo'); $paymentOption->setAdditionalInformation($this->initVenmoButton()->render()); $paymentOption->setLogo(Media::getMediaPath(_PS_MODULE_DIR_ . $this->name . '/views/img/paypal_logo.png')); return $paymentOption; } protected function initVenmoButton() { return new VenmoButton(); } protected function renderSepaOption($params) { $paymentOption = new PaymentOption(); $paymentOption->setModuleName('paypal_sepa'); $paymentOption->setCallToActionText($this->l('SEPA')); $paymentOption->setAction( sprintf( 'javascript:alert(\'%s\');', $this->l('Should use the SEPS payment button') ) ); $paymentOption->setAdditionalInformation($this->initSepaButton()->render()); return $paymentOption; } protected function initSepaButton() { return new SepaButton(); } public function renderPuiOption($params) { $paymentOption = new PaymentOption(); $action_text = $this->l('Pay upon invoice with Paypal'); $paymentOption->setLogo(Media::getMediaPath(_PS_MODULE_DIR_ . $this->name . '/views/img/paypal_logo.png')); $paymentOption->setModuleName('paypal_pui'); $paymentOption->setCallToActionText($action_text); $paymentOption->setAdditionalInformation($this->getFraudNetForm()->render()); return $paymentOption; } protected function getFraudSessionId() { return new FraudSessionId(); } /** @return FraudNetForm*/ public function getFraudNetForm() { return new FraudNetForm(); } protected function initApmFunctionality() { return new ApmFunctionality(); } protected function buildApmPaymentOptions($params) { $paymentOptions = []; $optionsMap = $this->buildOptionsMapAccordingToContext(); foreach ($optionsMap as $optionMap) { $paymentOption = new PaymentOption(); $paymentOption->setCallToActionText($optionMap['label']); $paymentOption->setAction( sprintf( 'javascript:alert(\'%s\');', $this->l('Should use the alternative payment button') // todo: specify message ) ); $paymentOption->setModuleName('paypal_' . $optionMap['method']); $paymentOption->setAdditionalInformation($this->initApmCollection([$optionMap['method']])->render()); $paymentOptions[] = $paymentOption; } return $paymentOptions; } protected function initApmCollection($method = null) { return new ApmCollection($method); } protected function initAcdcFunctionality() { return new AcdcFunctionality(); } protected function buildAcdcPaymentOption($params) { $paymentOption = new PaymentOption(); $paymentOption->setCallToActionText($this->l('Credit/Debit card')); $paymentOption->setAction( sprintf( 'javascript:alert(\'%s\');', $this->l('Should use the ACDC button') // todo: specify message ) ); $paymentOption->setModuleName('paypal_acdc'); $paymentOption->setAdditionalInformation($this->initAcdcPaymentMethod()->render()); $paymentOption->setLogo(Media::getMediaPath(_PS_MODULE_DIR_ . $this->name . '/views/img/paypal_logo.png')); return $paymentOption; } protected function initAcdcPaymentMethod() { return new AcdcPaymentMethod(); } protected function buildPaypalWallet($params) { $paymentOption = new PaymentOption(); $paymentOption->setCallToActionText($this->l('PayPal')); $paymentOption->setAction( sprintf( 'javascript:alert(\'%s\');', $this->l('Should use the PayPal wallet button ') // todo: specify message ) ); $paymentOption->setModuleName($this->name); $paymentOption->setAdditionalInformation($this->getShortcutPaymentStep()->render()); $paymentOption->setLogo(Media::getMediaPath(_PS_MODULE_DIR_ . $this->name . '/views/img/paypal_logo.png')); return $paymentOption; } /** * @param $params * * @return array of the PaymentOption objects * * @throws Exception * @throws SmartyException */ public function renderEcPaymentOptions($params) { $paymentOptions = []; $is_virtual = 0; $additionalInformation = ''; foreach ($params['cart']->getProducts() as $key => $product) { if ($product['is_virtual']) { $is_virtual = 1; break; } } $paymentOption = new PaymentOption(); $action_text = $this->l('Pay with Paypal'); $paymentOption->setLogo(Media::getMediaPath(_PS_MODULE_DIR_ . $this->name . '/views/img/paypal_logo.png')); $paymentOption->setModuleName($this->name); if (Configuration::get('PAYPAL_API_ADVANTAGES')) { $action_text .= ' | ' . $this->l('It\'s simple, fast and secure'); } $this->context->smarty->assign([ 'path' => $this->_path, ]); $paymentOption->setCallToActionText($action_text); if (Configuration::get('PAYPAL_EXPRESS_CHECKOUT_IN_CONTEXT')) { $additionalInformation .= $this->getShortcutPaymentStep()->render(); } else { $paymentOption->setAction($this->context->link->getModuleLink($this->name, 'ecInit', ['credit_card' => '0'], true)); } if (!$is_virtual && Configuration::get('PAYPAL_API_ADVANTAGES')) { $additionalInformation .= $this->context->smarty->fetch('module:paypal/views/templates/front/payment_infos.tpl'); } $paymentOption->setAdditionalInformation($additionalInformation); $paymentOptions[] = $paymentOption; if ((Configuration::get('PAYPAL_EXPRESS_CHECKOUT_SHORTCUT') || Configuration::get('PAYPAL_EXPRESS_CHECKOUT_SHORTCUT_CART') || Configuration::get('PAYPAL_EXPRESS_CHECKOUT_SHORTCUT_SIGNUP')) && isset($this->context->cookie->paypal_ecs)) { $paymentOption = new PaymentOption(); $action_text = $this->l('Pay with paypal express checkout'); $paymentOption->setCallToActionText($action_text); $paymentOption->setModuleName('express_checkout_schortcut'); $paymentOption->setAction($this->context->link->getModuleLink($this->name, 'ecValidation', ['short_cut' => '1', 'token' => $this->context->cookie->paypal_ecs], true)); $paymentOptions[] = $paymentOption; } return $paymentOptions; } public function hookHeader() { $returnContent = ''; $this->context->controller->registerStylesheet($this->name . '-fo', 'modules/' . $this->name . '/views/css/paypal_fo.css'); $resources = []; $method = AbstractMethodPaypal::load($this->paypal_method); if ((int) Configuration::get('PAYPAL_NEED_CHECK_CREDENTIALS')) { $method->checkCredentials(); Configuration::updateValue('PAYPAL_NEED_CHECK_CREDENTIALS', 0); } if (Tools::getValue('controller') == 'order') { if (!$this->checkActiveModule()) { return; } $method = AbstractMethodPaypal::load($this->paypal_method); if ($method->isConfigured() == false) { return false; } $this->context->controller->registerJavascript($this->name . '-paypal-info', 'modules/' . $this->name . '/views/js/paypal-info.js'); $resources[] = _MODULE_DIR_ . $this->name . '/views/js/paypal-info.js'; // Show Shortcut on signup page if need // if ps version is '1.7.6' and bigger than use native hook displayPersonalInformationTop if ($this->isShowShortcut() && !$this->context->customer->isLogged() && !$this->context->customer->is_guest) { if (version_compare(_PS_VERSION_, '1.7.6', '<') && ((bool) Configuration::get(ShortcutConfiguration::CUSTOMIZE_STYLE) === false || (int) Configuration::get(ShortcutConfiguration::DISPLAY_MODE_SIGNUP) == ShortcutConfiguration::DISPLAY_MODE_TYPE_HOOK)) { $Shortcut = new ShortcutSignup(); $returnContent .= $this->renderBnpl(['sourcePage' => ShortcutConfiguration::SOURCE_PAGE_SIGNUP]) . $Shortcut->render(); } $returnContent .= $this->context->smarty->fetch('module:paypal/views/templates/front/prefetch.tpl'); return $returnContent; } if ((Configuration::get(ShortcutConfiguration::SHOW_ON_PRODUCT_PAGE) || Configuration::get(ShortcutConfiguration::SHOW_ON_CART_PAGE) || Configuration::get(ShortcutConfiguration::SHOW_ON_SIGNUP_STEP)) && (isset($this->context->cookie->paypal_ecs) || isset($this->context->cookie->paypal_pSc))) { $this->context->controller->registerJavascript($this->name . '-paypal-ec-sc', 'modules/' . $this->name . '/views/js/shortcut_payment.js'); $resources[] = _MODULE_DIR_ . $this->name . '/views/js/shortcut_payment.js' . '?v=' . $this->version; if (isset($this->context->cookie->paypal_ecs)) { Media::addJsDef([ 'paypalCheckedMethod' => 'express_checkout_schortcut', ]); $cookie_paypal_email = $this->context->cookie->paypal_ecs_email; } elseif (isset($this->context->cookie->paypal_pSc)) { Media::addJsDef([ 'paypalCheckedMethod' => 'paypal_plus_schortcut', ]); $cookie_paypal_email = $this->context->cookie->paypal_pSc_email; } else { $cookie_paypal_email = ''; } $this->context->smarty->assign('paypalEmail', $cookie_paypal_email); $this->context->smarty->assign('isSandbox', $method->isSandbox()); $carrierFees = $this->context->cart->getOrderTotal(true, Cart::ONLY_SHIPPING); if ($carrierFees == 0) { $messageForCustomer = $this->context->smarty->fetch('module:paypal/views/templates/front/_partials/messageForCustomerOne.tpl'); } else { $this->context->smarty->assign('carrierFees', Tools::displayPrice($carrierFees)); $messageForCustomer = $this->context->smarty->fetch('module:paypal/views/templates/front/_partials/messageForCustomerTwo.tpl'); } Media::addJsDefL('scPaypalCheckedMsg', $messageForCustomer); } if ($this->paypal_method == 'MB') { $method->assignJSvarsPaypalMB(); $this->context->controller->registerJavascript($this->name . '-plusdcc-minjs', 'https://www.paypalobjects.com/webstatic/ppplusdcc/ppplusdcc.min.js', ['server' => 'remote']); $this->context->controller->registerJavascript($this->name . '-mb-payment-js', 'modules/' . $this->name . '/views/js/payment_mb.js'); $resources[] = _MODULE_DIR_ . $this->name . '/views/js/payment_mb.js' . '?v=' . $this->version; $resources[] = 'https://www.paypalobjects.com/webstatic/ppplusdcc/ppplusdcc.min.js' . '?v=' . $this->version; } } elseif (Tools::getValue('controller') == 'cart') { if (!$this->checkActiveModule()) { return; } if ($this->paypal_method == 'MB') { $resources[] = 'https://www.paypalobjects.com/webstatic/ppplusdcc/ppplusdcc.min.js' . '?v=' . $this->version; } } $this->context->smarty->assign('resources', $resources); $returnContent .= $this->context->smarty->fetch('module:paypal/views/templates/front/prefetch.tpl'); return $returnContent; } public function hookDisplayInvoiceLegalFreeText($params) { $paypal_order = PaypalOrder::loadByOrderId($params['order']->id); if (!Validate::isLoadedObject($paypal_order) || $paypal_order->method != 'PPP' || $paypal_order->payment_tool != 'PAY_UPON_INVOICE') { return ''; } $method = AbstractMethodPaypal::load(); $response = $method->getInfo($paypal_order->id_payment); if ($response->isSuccess() == false) { return ''; } $bankDetails = $response->getDepositBankDetails(); $tab = $this->l('The bank name') . ' : ' . $bankDetails->getBankName() . '; ' . $this->l('Account holder name') . ' : ' . $bankDetails->getAccountHolderName() . '; ' . $this->l('IBAN') . ' : ' . $bankDetails->getIban() . '; ' . $this->l('BIC') . ' : ' . $bankDetails->getBic(); return $tab; } public function checkActiveModule() { $active = false; $modules = Hook::getHookModuleExecList('paymentOptions'); if (empty($modules)) { return; } foreach ($modules as $module) { if ($module['module'] == $this->name) { $active = true; } } return $active; } /** * Get url for BT onboarding * * @param object $ps_order PS order object * @param string $transaction_id payment transaction ID */ public function setTransactionId($ps_order, $transaction_id) { Db::getInstance()->update('order_payment', [ 'transaction_id' => pSQL($transaction_id), ], 'order_reference = "' . pSQL($ps_order->reference) . '"'); } public function hookDisplayOrderConfirmation($params) { $paypal_order = PaypalOrder::loadByOrderId($params['order']->id); if (!Validate::isLoadedObject($paypal_order)) { return; } $this->context->smarty->assign([ 'transaction_id' => $paypal_order->id_transaction, 'method' => $paypal_order->method, ]); $this->context->controller->registerJavascript($this->name . '-order_confirmation_js', $this->_path . '/views/js/order_confirmation.js'); return $this->context->smarty->fetch('module:paypal/views/templates/hook/order_confirmation.tpl'); } public function hookDisplayOrderPreview($params) { $params['class_logger'] = 'PaypalLog'; if ($result = $this->handleExtensionsHook(__FUNCTION__, $params)) { if (!is_null($result)) { return $result; } } } public function hookDisplayReassurance() { if ($this->context->controller instanceof ProductController) { $content = ''; $bannerManager = $this->getBannerManager(); if ($bannerManager->isBannerAvailable() && version_compare(_PS_VERSION_, '1.7.6', '<')) { $content .= $bannerManager->renderForProductPage(); } $content .= $this->displayShortcutButton([ 'sourcePage' => ShortcutConfiguration::SOURCE_PAGE_PRODUCT, 'hook' => ShortcutConfiguration::HOOK_REASSURANCE, ]); if (version_compare(_PS_VERSION_, '1.7.6', '<')) { $content .= $this->renderBnpl(['sourcePage' => ShortcutConfiguration::SOURCE_PAGE_PRODUCT]); } return $content; } if ($this->context->controller instanceof CartController) { return $this->displayShortcutButton([ 'sourcePage' => ShortcutConfiguration::SOURCE_PAGE_CART, 'hook' => ShortcutConfiguration::HOOK_REASSURANCE, ]); } if ($this->context->controller instanceof OrderController) { $bannerManager = $this->getBannerManager(); if ($bannerManager->isBannerAvailable()) { return $bannerManager->renderForCheckoutPage(); } } return ''; } /** * @param array $data * * @return string */ public function renderBnpl($data) { $bnplOption = new BNPLOption(); $bnplAvailabilityManager = $this->getBnplAvailabilityManager(); $bnpl = new BNPLDummy(); if (false == $bnplAvailabilityManager->isEligibleCountryConfiguration()) { return ''; } if ($bnplOption->isEnable() == false) { return ''; } if (false == $bnplAvailabilityManager->isEligibleContext()) { return ''; } if ($data['sourcePage'] == ConfigurationMap::PAGE_TYPE_PAYMENT_STEP) { if ($bnplOption->displayOnPaymentStep() == false) { return ''; } $bnpl = new BNPLPaymentStep(); } if ($data['sourcePage'] == ShortcutConfiguration::SOURCE_PAGE_CART) { if ($bnplOption->displayOnCart() == false) { return ''; } if ($this->context->cart->nbProducts() == 0 || $this->context->cart->checkQuantities() === false) { return ''; } $bnpl = new BNPLCart(); } if ($data['sourcePage'] == ShortcutConfiguration::SOURCE_PAGE_PRODUCT) { if ($bnplOption->displayOnProduct() == false) { return ''; } $bnpl = new BNPLProduct((int) Tools::getValue('id_product')); } if ($data['sourcePage'] == ShortcutConfiguration::SOURCE_PAGE_SIGNUP) { if ($bnplOption->displayOnSignup() == false) { return ''; } $bnpl = new BNPLSignup(); } if ($this->paypal_method == 'MB') { $methodType = 'EC'; } else { $methodType = $this->paypal_method; } $method = AbstractMethodPaypal::load($methodType); if ($method->isConfigured() == false) { return ''; } return $bnpl->render(); } /** * @param array $data * * @return string */ public function displayShortcutButton($data) { if ($this->isShowShortcut() === false) { return ''; } if (false === isset($data['sourcePage'])) { return ''; } if (isset($data['hook'])) { if ($data['sourcePage'] == ShortcutConfiguration::SOURCE_PAGE_PRODUCT) { if (Configuration::get(ShortcutConfiguration::CUSTOMIZE_STYLE) && (int) Configuration::get(ShortcutConfiguration::DISPLAY_MODE_PRODUCT) !== ShortcutConfiguration::DISPLAY_MODE_TYPE_HOOK) { return ''; } // Take a hook by default if (version_compare(_PS_VERSION_, '1.7.6', '<') || (int) Configuration::getGlobalValue(ShortcutConfiguration::USE_OLD_HOOK)) { $hookSetted = ShortcutConfiguration::HOOK_REASSURANCE; } else { $hookSetted = ShortcutConfiguration::HOOK_PRODUCT_ACTIONS; } // If a style customization conf is active, take a hook configured if (Configuration::get(ShortcutConfiguration::CUSTOMIZE_STYLE)) { $hookSetted = Configuration::get(ShortcutConfiguration::PRODUCT_PAGE_HOOK); } if ($hookSetted != $data['hook']) { return ''; } } if ($data['sourcePage'] == ShortcutConfiguration::SOURCE_PAGE_CART) { if ($this->context->cart->nbProducts() == 0 || $this->context->cart->checkQuantities() === false) { return ''; } if (Configuration::get(ShortcutConfiguration::CUSTOMIZE_STYLE) && (int) Configuration::get(ShortcutConfiguration::DISPLAY_MODE_CART) !== ShortcutConfiguration::DISPLAY_MODE_TYPE_HOOK) { return ''; } // Take a hook by default if ((int) Configuration::getGlobalValue(ShortcutConfiguration::USE_OLD_HOOK)) { $hookSetted = ShortcutConfiguration::HOOK_SHOPPING_CART_FOOTER; } else { $hookSetted = ShortcutConfiguration::HOOK_EXPRESS_CHECKOUT; } // If a style customization conf is active, take a hook configured if (Configuration::get(ShortcutConfiguration::CUSTOMIZE_STYLE)) { $hookSetted = Configuration::get(ShortcutConfiguration::CART_PAGE_HOOK); } if ($hookSetted != $data['hook']) { return ''; } } if ($data['sourcePage'] == ShortcutConfiguration::SOURCE_PAGE_SIGNUP) { if (Configuration::get(ShortcutConfiguration::CUSTOMIZE_STYLE) && (int) Configuration::get(ShortcutConfiguration::DISPLAY_MODE_SIGNUP) !== ShortcutConfiguration::DISPLAY_MODE_TYPE_HOOK) { return ''; } } } if ($this->paypal_method == 'MB') { $methodType = 'EC'; } else { $methodType = $this->paypal_method; } $method = AbstractMethodPaypal::load($methodType); if ($method->isConfigured() == false) { return ''; } return $method->renderExpressCheckoutShortCut($data['sourcePage'], (isset($data['isWidget']) ? $data['isWidget'] : false)); } /** * Check if we need convert currency * * @return bool|int currency id */ public function needConvert() { $currency_mode = Currency::getPaymentCurrenciesSpecial($this->id); $mode_id = $currency_mode['id_currency']; if ($mode_id == -2) { return (int) Configuration::get('PS_CURRENCY_DEFAULT'); } elseif ($mode_id == -1) { return false; } elseif ($mode_id != $this->context->currency->id) { return (int) $mode_id; } else { return false; } } /** * Get payment currency iso code * * @return string currency iso code */ public function getPaymentCurrencyIso() { if ($id_currency = $this->needConvert()) { $currency = new Currency((int) $id_currency); } else { if (Validate::isLoadedObject(Context::getContext()->currency)) { $currency = Context::getContext()->currency; } else { $currency = new Currency((int) Configuration::get('PS_CURRENCY_DEFAULT')); } } return $currency->iso_code; } public function validateOrder($id_cart, $id_order_state, $amount_paid, $payment_method = 'Unknown', $message = null, $transaction = [], $currency_special = null, $dont_touch_amount = false, $secure_key = false, Shop $shop = null) { if ($this->needConvert()) { $amount_paid_curr = Tools::ps_round(Tools::convertPrice($amount_paid, new Currency($currency_special), true), 2); } else { $amount_paid_curr = Tools::ps_round($amount_paid, 2); } $amount_paid = Tools::ps_round($amount_paid, 2); $cart = new Cart((int) $id_cart); $total_ps = (float) $cart->getOrderTotal(true, Cart::BOTH); if ($amount_paid_curr > $total_ps + 0.10 || $amount_paid_curr < $total_ps - 0.10) { $total_ps = $amount_paid_curr; } try { parent::validateOrder( (int) $id_cart, (int) $id_order_state, (float) $total_ps, $payment_method, $message, ['transaction_id' => isset($transaction['transaction_id']) ? $transaction['transaction_id'] : ''], $currency_special, $dont_touch_amount, $secure_key, $shop ); } catch (Exception $e) { $log = 'Order validation error : ' . $e->getMessage() . ';'; $log .= ' File: ' . $e->getFile() . ';'; $log .= ' Line: ' . $e->getLine() . ';'; ProcessLoggerHandler::openLogger(); ProcessLoggerHandler::logError( $log, isset($transaction['transaction_id']) ? $transaction['transaction_id'] : null, null, (int) $id_cart, $this->context->shop->id, isset($transaction['payment_tool']) && $transaction['payment_tool'] ? $transaction['payment_tool'] : 'PayPal', (int) Configuration::get('PAYPAL_SANDBOX'), isset($transaction['date_transaction']) ? $transaction['date_transaction'] : null ); ProcessLoggerHandler::closeLogger(); $this->currentOrder = (int) Order::getIdByCartId((int) $id_cart); if ($this->currentOrder == false) { $msg = $this->l('Order validation error : ') . $e->getMessage() . '. '; if (isset($transaction['transaction_id']) && $id_order_state != Configuration::get('PS_OS_ERROR')) { $msg .= $this->l('Attention, your payment is made. Please, contact customer support. Your transaction ID is : ') . $transaction['transaction_id']; } Tools::redirect(Context::getContext()->link->getModuleLink('paypal', 'error', ['error_msg' => $msg, 'no_retry' => true])); } } $adminEmployee = new Employee(_PS_ADMIN_PROFILE_); $order = new Order($this->currentOrder); $orderState = new OrderState($order->current_state, $adminEmployee->id_lang); ProcessLoggerHandler::openLogger(); ProcessLoggerHandler::logInfo( $orderState->name, isset($transaction['transaction_id']) ? $transaction['transaction_id'] : null, $this->currentOrder, (int) $id_cart, $this->context->shop->id, isset($transaction['payment_tool']) && $transaction['payment_tool'] ? $transaction['payment_tool'] : 'PayPal', (int) Configuration::get('PAYPAL_SANDBOX'), isset($transaction['date_transaction']) ? $transaction['date_transaction'] : null ); ProcessLoggerHandler::closeLogger(); if (Tools::version_compare(_PS_VERSION_, '1.7.1.0', '>')) { $order = Order::getByCartId($id_cart); } else { $id_order = Order::getOrderByCartId($id_cart); $order = new Order($id_order); } if ($amount_paid_curr != 0 && $order->total_paid != $amount_paid_curr && $this->isOneOrder($order->reference)) { $order->total_paid = $amount_paid_curr; $order->total_paid_real = $amount_paid_curr; $order->total_paid_tax_incl = $amount_paid_curr; $order->update(); $sql = 'UPDATE `' . _DB_PREFIX_ . 'order_payment` SET `amount` = ' . (float) $amount_paid_curr . ' WHERE `order_reference` = "' . pSQL($order->reference) . '"'; Db::getInstance()->execute($sql); } //if there isn't a method, then we don't create PaypalOrder and PaypalCapture if (isset($transaction['method']) && $transaction['method']) { $paypal_order = new PaypalOrder(); $paypal_order->id_order = $this->currentOrder; $paypal_order->id_cart = $id_cart; $paypal_order->id_transaction = $transaction['transaction_id']; $paypal_order->id_payment = $transaction['id_payment']; $paypal_order->payment_method = $transaction['payment_method']; $paypal_order->currency = $transaction['currency']; $paypal_order->total_paid = (float) $amount_paid; $paypal_order->payment_status = $transaction['payment_status']; $paypal_order->total_prestashop = (float) $total_ps; $paypal_order->method = $transaction['method']; $paypal_order->payment_tool = isset($transaction['payment_tool']) ? $transaction['payment_tool'] : 'PayPal'; $paypal_order->sandbox = (int) Configuration::get('PAYPAL_SANDBOX'); $paypal_order->intent = $transaction['intent']; $paypal_order->save(); if ($transaction['capture']) { $paypal_capture = new PaypalCapture(); $paypal_capture->id_paypal_order = $paypal_order->id; $paypal_capture->save(); } if ($this->getWebhookOption()->isEnable() && $this->getWebhookOption()->isAvailable()) { if (PaypalPaymentMode::isSale()) { $this ->getWebhookService() ->createForOrder( $paypal_order, $this->getStatusMapping()->getAcceptedStatus() ); } } } } public function hookDisplayAdminOrder($params) { // Since Ps 1.7.7 this hook is displayed at bottom of a page and we should use a hook DisplayAdminOrderTop if (version_compare(_PS_VERSION_, '1.7.7', '>=')) { return false; } $return = $this->getAdminOrderPageMessages($params); $return .= $this->getPartialRefund($params); return $return; } protected function getPartialRefund($params) { $paypal_order = PaypalOrder::loadByOrderId($params['id_order']); if (!Validate::isLoadedObject($paypal_order)) { return ''; } $this->context->smarty->assign('chb_paypal_refund', $this->l('Refund on PayPal')); return $this->context->smarty->fetch(_PS_MODULE_DIR_ . $this->name . '/views/templates/hook/partialRefund.tpl'); } public function hookDisplayAdminOrderTop($params) { $return = $this->getAdminOrderPageMessages($params); $return .= $this->getPartialRefund($params); return $return; } protected function getAdminOrderPageMessages($params) { /* @var $paypal_order PaypalOrder */ $id_order = $params['id_order']; $order = new Order((int) $id_order); $paypal_msg = "
"; $tmpMessage .= $this->l('This order has been paid via Braintree payment solution provided by PayPal module prior v5.0. ') . ''; $tmpMessage .= $this->l('Starting from v5.0.0 of PayPal module, Braintree payment solution won\'t be available via PayPal module anymore. You can continue using Braintree by installing the new Braintree module available via ') . "" . $this->l('addons.prestashop') . '' . ''; $tmpMessage .= $this->l('All actions on this order will not be processed by Braintree until you install the new module (ex: you cannot refund this order automatically by changing order status).'); $tmpMessage .= '
'; $paypal_msg .= $this->displayWarning($tmpMessage); } if ($paypal_order->sandbox) { $tmpMessage = ""; $tmpMessage .= $this->l('[SANDBOX] Please pay attention that payment for this order was made via PayPal Sandbox mode.'); $tmpMessage .= '
'; $paypal_msg .= $this->displayWarning($tmpMessage); } if (isset($_SESSION['need_refund']) && $_SESSION['need_refund']) { unset($_SESSION['need_refund']); $tmpMessage = ""; $tmpMessage .= $this->l('The order should be refunded before the cancellation. Please select the status "Refunded".'); $tmpMessage .= $this->l('You can cancel the order after. If you don\'t want to generate a refund automatically on PayPal when you change the status, you can disable it via the module settings: "Experience -> Advanced settings - Customize order status", select "no action".'); $tmpMessage .= '
'; $paypal_msg .= $this->displayWarning($tmpMessage); } if (isset($_SESSION['not_payed_capture']) && $_SESSION['not_payed_capture']) { unset($_SESSION['not_payed_capture']); $paypal_msg .= $this->displayWarning( '' . $this->l('You can\'t refund order as it hasn\'t be paid yet.') . '
' ); } if (Tools::getValue('error_refund')) { $paypal_msg .= $this->displayWarning( '' . $this->l('We encountered an unexpected problem during refund operation. For more details please see the \'PayPal\' tab in the order details.') . '
' ); } if (Tools::getValue('cancel_failed')) { $paypal_msg .= $this->displayWarning( '' . $this->l('We encountered an unexpected problem during cancel operation. For more details please see the \'PayPal\' tab in the order details.') . '
' ); } if ($order->current_state == Configuration::get('PS_OS_REFUND') && $paypal_order->payment_status == 'Refunded') { $msg = $this->l('Your order is fully refunded by PayPal.'); $paypal_msg .= $this->displayWarning( '' . $msg . '
' ); } if ($order->getCurrentOrderState()->paid == 1 && Validate::isLoadedObject($paypal_capture) && $paypal_capture->id_capture) { $msg = $this->l('Your order is captured by PayPal.'); $paypal_msg .= $this->displayWarning( '' . $msg . '
' ); } if (Tools::getValue('error_capture')) { $paypal_msg .= $this->displayWarning( '' . $this->l('We encountered an unexpected problem during capture operation. See messages for more details.') . '
' ); } if ($paypal_order->total_paid != $paypal_order->total_prestashop) { $preferences = $this->context->link->getAdminLink('AdminPreferences', true); $paypal_msg .= $this->displayWarning('' . $this->l('Product pricing has been modified as your rounding settings aren\'t compliant with PayPal.') . ' ' . $this->l('To avoid automatic rounding to customer for PayPal payments, please update your rounding settings.') . ' ' . '' . $this->l('Read more.') . '
'); } if (isset($_SESSION['paypal_transaction_already_refunded']) && $_SESSION['paypal_transaction_already_refunded']) { unset($_SESSION['paypal_transaction_already_refunded']); $tmpMessage = ''; $tmpMessage .= $this->l('The order status was changed but this transaction has already been fully refunded.'); $tmpMessage .= '
'; $paypal_msg .= $this->displayWarning($tmpMessage); } if (isset($_SESSION['paypal_partial_refund_successful']) && $_SESSION['paypal_partial_refund_successful']) { unset($_SESSION['paypal_partial_refund_successful']); $tmpMessage = ''; $tmpMessage .= $this->l('A refund request has been sent to PayPal.'); $tmpMessage .= '
'; $paypal_msg .= $this->displayWarning($tmpMessage); } $paypal_msg .= '