mirror of
https://gh.wpcy.net/https://github.com/woocommerce/woocommerce-paypal-payments.git
synced 2026-05-04 08:13:02 +08:00
353 lines
15 KiB
PHP
353 lines
15 KiB
PHP
<?php
|
|
|
|
/**
|
|
* The Axo module.
|
|
*
|
|
* @package WooCommerce\PayPalCommerce\Axo
|
|
*/
|
|
declare (strict_types=1);
|
|
namespace WooCommerce\PayPalCommerce\Axo;
|
|
|
|
use WooCommerce\PayPalCommerce\Assets\AssetGetter;
|
|
use WooCommerce\PayPalCommerce\Axo\Assets\AxoManager;
|
|
use WooCommerce\PayPalCommerce\Axo\Endpoint\AxoScriptAttributes;
|
|
use WooCommerce\PayPalCommerce\Axo\Endpoint\FrontendLogger;
|
|
use WooCommerce\PayPalCommerce\Axo\Gateway\AxoGateway;
|
|
use WooCommerce\PayPalCommerce\Axo\Service\AxoApplies;
|
|
use WooCommerce\PayPalCommerce\Button\Assets\SmartButtonInterface;
|
|
use WooCommerce\PayPalCommerce\Button\Helper\Context;
|
|
use WooCommerce\PayPalCommerce\Session\SessionHandler;
|
|
use WooCommerce\PayPalCommerce\Settings\Data\SettingsProvider;
|
|
use WooCommerce\PayPalCommerce\Vendor\Inpsyde\Modularity\Module\ExecutableModule;
|
|
use WooCommerce\PayPalCommerce\Vendor\Inpsyde\Modularity\Module\ModuleClassNameIdTrait;
|
|
use WooCommerce\PayPalCommerce\Vendor\Inpsyde\Modularity\Module\ServiceModule;
|
|
use WooCommerce\PayPalCommerce\Vendor\Psr\Container\ContainerInterface;
|
|
use WooCommerce\PayPalCommerce\WcGateway\Gateway\CreditCardGateway;
|
|
use WooCommerce\PayPalCommerce\WcGateway\Gateway\PayPalGateway;
|
|
use WooCommerce\PayPalCommerce\WcSubscriptions\Helper\SubscriptionHelper;
|
|
use WC_Payment_Gateways;
|
|
use WooCommerce\PayPalCommerce\WcGateway\Helper\CardPaymentsConfiguration;
|
|
/**
|
|
* Class AxoModule
|
|
*
|
|
* @psalm-suppress MissingConstructor
|
|
*/
|
|
class AxoModule implements ServiceModule, ExecutableModule
|
|
{
|
|
use ModuleClassNameIdTrait;
|
|
/**
|
|
* The session handler for ContextTrait.
|
|
*
|
|
* @var SessionHandler|null
|
|
*/
|
|
protected ?SessionHandler $session_handler;
|
|
/**
|
|
* {@inheritDoc}
|
|
*/
|
|
public function services(): array
|
|
{
|
|
return require __DIR__ . '/../services.php';
|
|
}
|
|
/**
|
|
* {@inheritDoc}
|
|
*/
|
|
public function run(ContainerInterface $c): bool
|
|
{
|
|
add_filter(
|
|
'woocommerce_payment_gateways',
|
|
/**
|
|
* Param types removed to avoid third-party issues.
|
|
*
|
|
* @psalm-suppress MissingClosureParamType
|
|
*/
|
|
function ($methods) use ($c) {
|
|
if (!is_array($methods)) {
|
|
return $methods;
|
|
}
|
|
$gateway = $c->get('axo.gateway');
|
|
// Check if the module is applicable, correct country, currency, ... etc.
|
|
if (!$c->get('axo.eligible')) {
|
|
return $methods;
|
|
}
|
|
// Add the gateway in admin area.
|
|
if (is_admin()) {
|
|
/**
|
|
* @var Context $context
|
|
*/
|
|
$context = $c->get('button.helper.context');
|
|
if (!$context->is_wc_settings_payments_tab()) {
|
|
$methods[] = $gateway;
|
|
}
|
|
return $methods;
|
|
}
|
|
if (is_user_logged_in()) {
|
|
return $methods;
|
|
}
|
|
$dcc_configuration = $c->get('wcgateway.configuration.card-configuration');
|
|
assert($dcc_configuration instanceof CardPaymentsConfiguration);
|
|
if (!$dcc_configuration->is_enabled()) {
|
|
return $methods;
|
|
}
|
|
if ($this->is_excluded_endpoint()) {
|
|
return $methods;
|
|
}
|
|
$methods[] = $gateway;
|
|
return $methods;
|
|
},
|
|
1,
|
|
9
|
|
);
|
|
// Hides credit card gateway on checkout when using Fastlane.
|
|
add_filter(
|
|
'woocommerce_available_payment_gateways',
|
|
/**
|
|
* Param types removed to avoid third-party issues.
|
|
*
|
|
* @psalm-suppress MissingClosureParamType
|
|
*/
|
|
function ($methods) use ($c): array {
|
|
if (!is_array($methods) || !$c->get('axo.eligible')) {
|
|
return $methods;
|
|
}
|
|
if (apply_filters('woocommerce_paypal_payments_axo_hide_credit_card_gateway', $this->hide_credit_card_when_using_fastlane($methods, $c))) {
|
|
unset($methods[CreditCardGateway::ID]);
|
|
}
|
|
return $methods;
|
|
}
|
|
);
|
|
// Enforce Fastlane to always be the first payment method in the list.
|
|
add_action('wc_payment_gateways_initialized', function (WC_Payment_Gateways $gateways) {
|
|
if (is_admin()) {
|
|
return;
|
|
}
|
|
foreach ($gateways->payment_gateways as $key => $gateway) {
|
|
if ($gateway->id === AxoGateway::ID) {
|
|
unset($gateways->payment_gateways[$key]);
|
|
array_unshift($gateways->payment_gateways, $gateway);
|
|
break;
|
|
}
|
|
}
|
|
});
|
|
add_action('wp_loaded', function () use ($c) {
|
|
$this->session_handler = $c->get('session.handler');
|
|
$settings_provider = $c->get('settings.settings-provider');
|
|
assert($settings_provider instanceof SettingsProvider);
|
|
$is_paypal_enabled = $settings_provider->is_method_enabled(PayPalGateway::ID);
|
|
$subscription_helper = $c->get('wc-subscriptions.helper');
|
|
assert($subscription_helper instanceof SubscriptionHelper);
|
|
/**
|
|
* @var Context $context
|
|
*/
|
|
$context = $c->get('button.helper.context');
|
|
// Check if the module is applicable, correct country, currency, ... etc.
|
|
if (!$is_paypal_enabled || !$c->get('axo.eligible') || $context->is_paypal_continuation() || $subscription_helper->cart_contains_subscription()) {
|
|
return;
|
|
}
|
|
$manager = $c->get('axo.manager');
|
|
assert($manager instanceof AxoManager);
|
|
// Enqueue frontend scripts.
|
|
add_action('wp_enqueue_scripts', function () use ($c, $manager) {
|
|
$smart_button = $c->get('button.smart-button');
|
|
assert($smart_button instanceof SmartButtonInterface);
|
|
$axo_applies = $c->get('axo.service.axo-applies');
|
|
assert($axo_applies instanceof AxoApplies);
|
|
if ($axo_applies->should_render_fastlane() && $smart_button->should_load_ppcp_script()) {
|
|
$manager->enqueue();
|
|
}
|
|
});
|
|
// Render submit button.
|
|
add_action($manager->checkout_button_renderer_hook(), function () use ($c, $manager) {
|
|
$axo_applies = $c->get('axo.service.axo-applies');
|
|
assert($axo_applies instanceof AxoApplies);
|
|
if ($axo_applies->should_render_fastlane()) {
|
|
$manager->render_checkout_button();
|
|
}
|
|
});
|
|
/**
|
|
* Late loading locations because of trouble with some shipping plugins
|
|
*/
|
|
add_filter('woocommerce_paypal_payments_axo_shipping_wc_enabled_locations', function (array $locations) use ($c): array {
|
|
return array_merge($locations, $c->get('axo.shipping-wc-enabled-locations'));
|
|
});
|
|
/**
|
|
* Param types removed to avoid third-party issues.
|
|
*
|
|
* @psalm-suppress MissingClosureParamType
|
|
*/
|
|
add_filter('woocommerce_paypal_payments_sdk_components_hook', function ($components) use ($c) {
|
|
$dcc_configuration = $c->get('wcgateway.configuration.card-configuration');
|
|
assert($dcc_configuration instanceof CardPaymentsConfiguration);
|
|
if (!$dcc_configuration->use_fastlane()) {
|
|
return $components;
|
|
}
|
|
$components[] = 'fastlane';
|
|
return $components;
|
|
});
|
|
add_action('wp_head', function () use ($c) {
|
|
// Add meta tag to allow feature-detection of the site's AXO payment state.
|
|
$dcc_configuration = $c->get('wcgateway.configuration.card-configuration');
|
|
assert($dcc_configuration instanceof CardPaymentsConfiguration);
|
|
if ($dcc_configuration->use_fastlane()) {
|
|
// phpcs:ignore WordPress.WP.EnqueuedResources.NonEnqueuedScript
|
|
echo '<script async src="https://www.paypalobjects.com/insights/v1/paypal-insights.sandbox.min.js"></script>';
|
|
$this->add_feature_detection_tag(\true);
|
|
} else {
|
|
$this->add_feature_detection_tag(\false);
|
|
}
|
|
});
|
|
// Set Axo as the default payment method on checkout for guest customers.
|
|
add_action('template_redirect', function () use ($c) {
|
|
$axo_applies = $c->get('axo.service.axo-applies');
|
|
assert($axo_applies instanceof AxoApplies);
|
|
if ($axo_applies->should_render_fastlane()) {
|
|
WC()->session->set('chosen_payment_method', AxoGateway::ID);
|
|
}
|
|
});
|
|
// Add the markup necessary for displaying overlays and loaders for Axo on the checkout page.
|
|
$this->add_checkout_loader_markup($c);
|
|
}, 1);
|
|
add_action('wc_ajax_' . FrontendLogger::ENDPOINT, static function () use ($c) {
|
|
$endpoint = $c->get('axo.endpoint.frontend-logger');
|
|
assert($endpoint instanceof FrontendLogger);
|
|
$endpoint->handle_request();
|
|
});
|
|
// Enqueue the PayPal Insights script.
|
|
add_action('wp_enqueue_scripts', function () use ($c) {
|
|
$this->enqueue_paypal_insights_script_on_order_received($c);
|
|
});
|
|
add_filter(
|
|
'ppcp_return_url_error_args',
|
|
/**
|
|
* Param types removed to avoid third-party issues.
|
|
*
|
|
* @psalm-suppress MissingClosureParamType
|
|
*/
|
|
function ($args) use ($c): array {
|
|
$axo_applies = $c->get('axo.service.axo-applies');
|
|
assert($axo_applies instanceof AxoApplies);
|
|
if ($axo_applies->should_render_fastlane()) {
|
|
$args['ppcp_fastlane_error'] = '1';
|
|
}
|
|
return $args;
|
|
}
|
|
);
|
|
// Remove Fastlane on the Pay for Order page.
|
|
add_filter(
|
|
'woocommerce_available_payment_gateways',
|
|
/**
|
|
* Param types removed to avoid third-party issues.
|
|
*
|
|
* @psalm-suppress MissingClosureParamType
|
|
*/
|
|
static function ($methods) {
|
|
if (!is_array($methods) || !is_wc_endpoint_url('order-pay')) {
|
|
return $methods;
|
|
}
|
|
// Remove Fastlane if present.
|
|
unset($methods[AxoGateway::ID]);
|
|
return $methods;
|
|
}
|
|
);
|
|
add_action('wc_ajax_' . AxoScriptAttributes::ENDPOINT, static function () use ($c) {
|
|
$endpoint = $c->get('axo.endpoint.script-attributes');
|
|
assert($endpoint instanceof AxoScriptAttributes);
|
|
$endpoint->handle_request();
|
|
});
|
|
return \true;
|
|
}
|
|
/**
|
|
* Condition to evaluate if Credit Card gateway should be hidden.
|
|
*
|
|
* @param array $methods WC payment methods.
|
|
* @param ContainerInterface $c The container.
|
|
* @return bool
|
|
*/
|
|
private function hide_credit_card_when_using_fastlane(array $methods, ContainerInterface $c): bool
|
|
{
|
|
$axo_applies = $c->get('axo.service.axo-applies');
|
|
assert($axo_applies instanceof AxoApplies);
|
|
return $axo_applies->should_render_fastlane() && isset($methods[CreditCardGateway::ID]);
|
|
}
|
|
/**
|
|
* Adds the markup necessary for displaying overlays and loaders for Axo on the checkout page.
|
|
*
|
|
* @param ContainerInterface $c The container.
|
|
* @return void
|
|
*/
|
|
private function add_checkout_loader_markup(ContainerInterface $c): void
|
|
{
|
|
$axo_applies = $c->get('axo.service.axo-applies');
|
|
assert($axo_applies instanceof AxoApplies);
|
|
if ($axo_applies->should_render_fastlane()) {
|
|
add_action('woocommerce_checkout_before_customer_details', function () {
|
|
echo '<div class="ppcp-axo-loading">';
|
|
});
|
|
add_action('woocommerce_checkout_after_customer_details', function () {
|
|
echo '</div>';
|
|
});
|
|
add_action('woocommerce_checkout_billing', function () {
|
|
echo '<div class="loader"><div class="ppcp-axo-overlay"></div>';
|
|
}, 8);
|
|
add_action('woocommerce_checkout_billing', function () {
|
|
echo '</div>';
|
|
}, 12);
|
|
}
|
|
}
|
|
/**
|
|
* Condition to evaluate if the current endpoint is excluded.
|
|
*
|
|
* @return bool
|
|
*/
|
|
private function is_excluded_endpoint(): bool
|
|
{
|
|
return is_wc_endpoint_url('order-pay') || is_wc_endpoint_url('order-received');
|
|
}
|
|
/**
|
|
* Outputs a meta tag to allow feature detection on certain pages.
|
|
*
|
|
* @param bool $axo_enabled Whether the gateway is enabled.
|
|
* @return void
|
|
*/
|
|
private function add_feature_detection_tag(bool $axo_enabled)
|
|
{
|
|
$show_tag = is_home() || is_checkout() || is_cart() || is_shop();
|
|
if (!$show_tag) {
|
|
return;
|
|
}
|
|
printf('<meta name="ppcp.axo" content="ppcp.axo.%s" />', $axo_enabled ? 'enabled' : 'disabled');
|
|
}
|
|
/**
|
|
* Enqueues PayPal Insights on the Order Received endpoint.
|
|
*
|
|
* @param ContainerInterface $c The service container.
|
|
* @return void
|
|
*/
|
|
private function enqueue_paypal_insights_script_on_order_received(ContainerInterface $c): void
|
|
{
|
|
global $wp;
|
|
if (!isset($wp->query_vars['order-received'])) {
|
|
return;
|
|
}
|
|
$order_id = absint($wp->query_vars['order-received']);
|
|
if (!$order_id) {
|
|
return;
|
|
}
|
|
$order = wc_get_order($order_id);
|
|
if (!$order || !$order instanceof \WC_Order) {
|
|
return;
|
|
}
|
|
//phpcs:ignore WordPress.Security.NonceVerification.Recommended
|
|
$order_key_from_url = isset($_GET['key']) ? wc_clean(wp_unslash($_GET['key'])) : '';
|
|
//phpcs:ignore WordPress.WP.Capabilities.Unknown
|
|
if ($order->get_order_key() !== $order_key_from_url && !current_user_can('view_order', $order_id)) {
|
|
return;
|
|
}
|
|
$asset_getter = $c->get('axo.asset_getter');
|
|
assert($asset_getter instanceof AssetGetter);
|
|
$asset_version = $c->get('ppcp.asset-version');
|
|
$insights_data = $c->get('axo.insights');
|
|
wp_register_script('wc-ppcp-paypal-insights-end-checkout', $asset_getter->get_asset_url('Insights/EndCheckoutTracker.js'), array('wp-plugins', 'wp-data', 'wp-element', 'wc-blocks-registry'), $asset_version, \true);
|
|
wp_localize_script('wc-ppcp-paypal-insights-end-checkout', 'wc_ppcp_axo_insights_data', array_merge($insights_data, array('orderId' => $order_id, 'orderTotal' => (string) $order->get_total(), 'orderCurrency' => (string) $order->get_currency(), 'paymentMethod' => (string) $order->get_payment_method())));
|
|
wp_enqueue_script('wc-ppcp-paypal-insights-end-checkout');
|
|
}
|
|
}
|