diff --git a/inc/settings/mollie_advanced_settings.php b/inc/settings/mollie_advanced_settings.php
index 31750d6b..e4026f49 100644
--- a/inc/settings/mollie_advanced_settings.php
+++ b/inc/settings/mollie_advanced_settings.php
@@ -199,6 +199,36 @@ class="mollie-settings-advanced-payment-desc-label button button-secondary butto
__('Clear now', 'mollie-payments-for-woocommerce')
) . ')',
],
+ [
+ 'id' => $pluginName . '_place_payment_onhold',
+ 'title' => __('Placing payments on Hold', 'mollie-payments-for-woocommerce'),
+ 'type' => 'select',
+ 'desc_tip' => true,
+ 'options' => [
+ 'immediate_capture' => __('Capture payments immediately', 'mollie-payments-for-woocommerce'),
+ 'later_capture' => __('Authorize payments for a later capture', 'mollie-payments-for-woocommerce'),
+ ],
+ 'default' => 'immediate_capture',
+ 'desc' => sprintf(
+ __(
+ 'Authorized payment can be captured or voided by changing the order status instead of doing it manually.',
+ 'mollie-payments-for-woocommerce'
+ )
+ ),
+ ],
+ [
+ 'id' => $pluginName . '_capture_or_void',
+ 'title' => __(
+ 'Capture or void on status change',
+ 'mollie-payments-for-woocommerce'
+ ),
+ 'type' => 'checkbox',
+ 'default' => 'no',
+ 'desc' => __(
+ 'Capture authorized payments automatically when setting the order status to Processing or Completed. Void the payment by setting the order status Canceled.',
+ 'mollie-payments-for-woocommerce'
+ ),
+ ],
[
'id' => $pluginName . '_sectionend',
'type' => 'sectionend',
diff --git a/languages/en_GB.pot b/languages/en_GB.pot
index ecfbc1ea..eb67e741 100644
--- a/languages/en_GB.pot
+++ b/languages/en_GB.pot
@@ -2,13 +2,14 @@
# This file is distributed under the GPLv2 or later.
msgid ""
msgstr ""
-"Project-Id-Version: Mollie Payments for WooCommerce 7.4.0\n"
-"Report-Msgid-Bugs-To: https://wordpress.org/support/plugin/WooCommerce\n"
+"Project-Id-Version: Mollie Payments for WooCommerce 7.4.1-beta\n"
+"Report-Msgid-Bugs-To: https://wordpress.org/support/plugin/html\n"
"Last-Translator: FULL NAME \n"
"Language-Team: LANGUAGE \n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
+"POT-Creation-Date: 2023-10-20T15:51:47+00:00\n"
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
"X-Generator: WP-CLI 2.8.1\n"
"X-Domain: mollie-payments-for-woocommerce\n"
@@ -508,7 +509,7 @@ msgstr ""
#: src/Buttons/ApplePayButton/AppleAjaxRequests.php:700
#: src/Buttons/PayPalButton/PayPalAjaxRequests.php:113
#: src/Buttons/PayPalButton/PayPalAjaxRequests.php:157
-#: src/Payment/PaymentService.php:716
+#: src/Payment/PaymentService.php:718
msgid "Could not create %s payment."
msgstr ""
@@ -520,85 +521,89 @@ msgstr ""
msgid "%1$sApple Pay Validation Error%2$s Check %3$sApple Server requirements page%4$s to fix it in order to make the Apple Pay button work"
msgstr ""
-#: src/Gateway/GatewayModule.php:655
+#: src/Gateway/GatewayModule.php:326
+msgid "Mollie Payment Details"
+msgstr ""
+
+#: src/Gateway/GatewayModule.php:682
msgid "Error processing %1$s payment, the %2$s field is required."
msgstr ""
-#: src/Gateway/GatewayModule.php:669
+#: src/Gateway/GatewayModule.php:696
msgid "Please enter your %1$s, this is required for %2$s payments"
msgstr ""
-#: src/Gateway/MolliePaymentGateway.php:269
+#: src/Gateway/MolliePaymentGateway.php:276
#: src/Settings/Page/MollieSettingsPage.php:314
msgid "Test mode enabled."
msgstr ""
#. translators: The surrounding %s's Will be replaced by a link to the global setting page
-#: src/Gateway/MolliePaymentGateway.php:274
+#: src/Gateway/MolliePaymentGateway.php:281
msgid "No API key provided. Please %1$sset you Mollie API key%2$s first."
msgstr ""
#. translators: Placeholder 1: payment method title. The surrounding %s's Will be replaced by a link to the Mollie profile
-#: src/Gateway/MolliePaymentGateway.php:289
+#: src/Gateway/MolliePaymentGateway.php:296
msgid "%1$s not enabled in your Mollie profile. You can enable it by editing your %2$sMollie profile%3$s."
msgstr ""
#. translators: Placeholder 1: WooCommerce currency, placeholder 2: Supported Mollie currencies
-#: src/Gateway/MolliePaymentGateway.php:304
+#: src/Gateway/MolliePaymentGateway.php:311
msgid "Current shop currency %1$s not supported by Mollie. Read more about %2$ssupported currencies and payment methods.%3$s "
msgstr ""
-#: src/Gateway/MolliePaymentGateway.php:575
+#: src/Gateway/MolliePaymentGateway.php:582
msgid "You have cancelled your payment. Please complete your order with a different payment method."
msgstr ""
-#: src/Gateway/MolliePaymentGateway.php:596
-#: src/Gateway/MolliePaymentGateway.php:610
+#: src/Gateway/MolliePaymentGateway.php:603
+#: src/Gateway/MolliePaymentGateway.php:617
msgid "Your payment was not successful. Please complete your order with a different payment method."
msgstr ""
-#: src/Gateway/MolliePaymentGateway.php:646
+#: src/Gateway/MolliePaymentGateway.php:653
msgid "Could not load order %s"
msgstr ""
-#: src/Gateway/MolliePaymentGateway.php:899
+#: src/Gateway/MolliePaymentGateway.php:918
msgid "Order cancelled"
msgstr ""
#. translators: Placeholder 1: payment method title, placeholder 2: payment ID
-#: src/Gateway/MolliePaymentGateway.php:933
+#: src/Gateway/MolliePaymentGateway.php:952
msgid "%1$s payment still pending (%2$s) but customer already returned to the store. Status should be updated automatically in the future, if it doesn't this might indicate a communication issue between the site and Mollie."
msgstr ""
-#: src/Gateway/MolliePaymentGateway.php:939
+#: src/Gateway/MolliePaymentGateway.php:958
#: src/Payment/MollieObject.php:682
#: src/Payment/MollieObject.php:714
#: src/Payment/MollieOrder.php:281
#: src/Payment/MollieOrder.php:338
-#: src/Payment/MollieOrder.php:382
-#: src/Payment/MollieOrder.php:463
-#: src/Payment/MollieOrder.php:534
-#: src/Payment/MollieOrder.php:877
-#: src/Payment/MollieOrderService.php:171
-#: src/Payment/MollieOrderService.php:437
-#: src/Payment/MollieOrderService.php:500
-#: src/Payment/MollieOrderService.php:714
+#: src/Payment/MollieOrder.php:389
+#: src/Payment/MollieOrder.php:470
+#: src/Payment/MollieOrder.php:541
+#: src/Payment/MollieOrder.php:884
+#: src/Payment/MollieOrderService.php:172
+#: src/Payment/MollieOrderService.php:438
+#: src/Payment/MollieOrderService.php:501
+#: src/Payment/MollieOrderService.php:715
#: src/Payment/MolliePayment.php:236
#: src/Payment/MolliePayment.php:323
#: src/Payment/MolliePayment.php:399
#: src/Payment/MolliePayment.php:423
-#: src/Payment/PaymentService.php:801
+#: src/Payment/PaymentService.php:803
#: src/Subscription/MollieSepaRecurringGateway.php:137
#: src/Subscription/MollieSepaRecurringGateway.php:204
#: src/Subscription/MollieSubscriptionGateway.php:458
msgid "test mode"
msgstr ""
-#: src/Gateway/MolliePaymentGateway.php:950
+#: src/Gateway/MolliePaymentGateway.php:969
msgid ", payment pending."
msgstr ""
-#: src/Gateway/MolliePaymentGateway.php:982
+#: src/Gateway/MolliePaymentGateway.php:1001
msgid "Your order has been cancelled."
msgstr ""
@@ -766,72 +771,72 @@ msgid "Order authorized using %1$s payment (%2$s). Set order to completed in Woo
msgstr ""
#. translators: Placeholder 1: payment method title, placeholder 2: payment ID
-#: src/Payment/MollieOrder.php:380
+#: src/Payment/MollieOrder.php:387
msgid "Order completed at Mollie for %1$s order (%2$s). At least one order line completed. Remember: Completed status for an order at Mollie is not the same as Completed status in WooCommerce!"
msgstr ""
#. translators: Placeholder 1: payment method title, placeholder 2: payment ID
-#: src/Payment/MollieOrder.php:461
+#: src/Payment/MollieOrder.php:468
msgid "%1$s order (%2$s) cancelled ."
msgstr ""
#. translators: Placeholder 1: payment method title, placeholder 2: payment ID
-#: src/Payment/MollieOrder.php:532
+#: src/Payment/MollieOrder.php:539
msgid "%1$s order expired (%2$s) but not cancelled because of another pending payment (%3$s)."
msgstr ""
-#: src/Payment/MollieOrder.php:627
+#: src/Payment/MollieOrder.php:634
msgctxt "Order note error"
msgid "The sum of refunds for all order lines is not identical to the refund amount, so this refund will be processed as a payment amount refund, not an order line refund."
msgstr ""
-#: src/Payment/MollieOrder.php:759
+#: src/Payment/MollieOrder.php:766
msgctxt "Order note error"
msgid "Can not refund order amount that has status %1$s at Mollie."
msgstr ""
-#: src/Payment/MollieOrder.php:781
+#: src/Payment/MollieOrder.php:788
msgid "Amount refund of %1$s%2$s refunded in WooCommerce and at Mollie.%3$s Refund ID: %4$s."
msgstr ""
#. translators: Placeholder 1: payment method title, placeholder 2: payment ID
-#: src/Payment/MollieOrder.php:872
+#: src/Payment/MollieOrder.php:879
msgid "%1$s order (%2$s) expired ."
msgstr ""
-#: src/Payment/MollieOrder.php:1095
+#: src/Payment/MollieOrder.php:1102
msgid "%1$sx %2$s cancelled for %3$s%4$s in WooCommerce and at Mollie."
msgstr ""
-#: src/Payment/MollieOrder.php:1119
+#: src/Payment/MollieOrder.php:1126
msgid "%1$sx %2$s refunded for %3$s%4$s in WooCommerce and at Mollie.%5$s Refund ID: %6$s."
msgstr ""
#. translators: Placeholder 1: payment method title, placeholder 2: payment status, placeholder 3: payment ID
-#: src/Payment/MollieOrderService.php:168
+#: src/Payment/MollieOrderService.php:169
msgid "%1$s payment %2$s (%3$s), not processed."
msgstr ""
-#: src/Payment/MollieOrderService.php:400
+#: src/Payment/MollieOrderService.php:401
msgid "New chargeback %s processed! Order note and order status updated."
msgstr ""
#. translators: Placeholder 1: payment method title, placeholder 2: payment ID
-#: src/Payment/MollieOrderService.php:432
+#: src/Payment/MollieOrderService.php:433
msgid "%1$s payment charged back via Mollie (%2$s). You will need to manually review the payment (and adjust product stocks if you use it)."
msgstr ""
#. translators: Placeholder 1: payment method title, placeholder 2: payment ID
-#: src/Payment/MollieOrderService.php:494
+#: src/Payment/MollieOrderService.php:495
msgid "%1$s payment charged back via Mollie (%2$s). Subscription status updated, please review (and adjust product stocks if you use it)."
msgstr ""
#. translators: Placeholder 1: payment method title, placeholder 2: payment ID
-#: src/Payment/MollieOrderService.php:701
+#: src/Payment/MollieOrderService.php:702
msgid "%1$s payment %2$s via Mollie (%3$s %4$s). You will need to manually review the payment (and adjust product stocks if you use it)."
msgstr ""
-#: src/Payment/MollieOrderService.php:814
+#: src/Payment/MollieOrderService.php:815
msgid "New refund %s processed in Mollie Dashboard! Order note added, but order not updated."
msgstr ""
@@ -930,35 +935,39 @@ msgctxt "Order note info"
msgid "Order could not be canceled at Mollie, because order status is "
msgstr ""
-#: src/Payment/PaymentService.php:622
+#: src/Payment/PaymentService.php:624
msgid "Subscription switch failed, no valid mandate found. Place a completely new order to change your subscription."
msgstr ""
-#: src/Payment/PaymentService.php:628
+#: src/Payment/PaymentService.php:630
msgid "Failed switching subscriptions, no valid mandate."
msgstr ""
-#: src/Payment/PaymentService.php:638
+#: src/Payment/PaymentService.php:640
msgid "Order completed internally because of an existing valid mandate at Mollie."
msgstr ""
-#: src/Payment/PaymentService.php:779
+#: src/Payment/PaymentService.php:781
#: src/Subscription/MollieSepaRecurringGateway.php:126
#: src/Subscription/MollieSubscriptionGateway.php:449
msgid "Awaiting payment confirmation."
msgstr ""
#. translators: Placeholder 1: Payment method title, placeholder 2: payment ID
-#: src/Payment/PaymentService.php:799
+#: src/Payment/PaymentService.php:801
#: src/Subscription/MollieSepaRecurringGateway.php:135
#: src/Subscription/MollieSubscriptionGateway.php:456
msgid "%1$s payment started (%2$s)."
msgstr ""
-#: src/Payment/PaymentService.php:869
+#: src/Payment/PaymentService.php:871
msgid "Payment failed due to: Mollie is out of service. Please try again later."
msgstr ""
+#: src/Payment/PaymentService.php:894
+msgid "Payment failed due to: The payment was declined due to suspected fraud."
+msgstr ""
+
#: src/Payment/RefundLineItemsBuilder.php:126
msgid "Mollie doesn't allow a partial refund of the full amount or quantity of at least one order line. Trying to process this as an amount refund instead."
msgstr ""
@@ -1149,6 +1158,7 @@ msgstr ""
#: src/PaymentMethods/Ideal.php:17
#: src/PaymentMethods/Ideal.php:55
#: src/PaymentMethods/Kbc.php:16
+#: src/PaymentMethods/Kbc.php:60
msgid "Select your bank"
msgstr ""
@@ -1274,6 +1284,10 @@ msgstr ""
msgid "If you disable this, a dropdown with various KBC/CBC banks will not be shown in the WooCommerce checkout, so users will select a KBC/CBC bank on the Mollie payment page after checkout."
msgstr ""
+#: src/PaymentMethods/Kbc.php:54
+msgid "This text will be displayed as the first option in the KBC/CBC issuers drop down, if nothing is entered, 'Select your bank' will be shown. Only if the above 'Show KBC/CBC banks dropdown' is enabled."
+msgstr ""
+
#: src/PaymentMethods/Klarna.php:13
msgid "Pay with Klarna"
msgstr ""
@@ -1838,7 +1852,7 @@ msgid "Initial order status"
msgstr ""
#: src/Settings/General/MollieGeneralSettings.php:292
-msgid "Some payment methods take longer than a few hours to complete. The initial order state is then set to '%1$s'. This ensures the order is not cancelled when the setting %2$s is used."
+msgid "Some payment methods take longer than a few hours to complete. The initial order state is then set to '%1$s'. This ensures the order is not cancelled when the setting %2$s is used. This will also prevent the order to be canceled when expired."
msgstr ""
#: src/Settings/Page/MollieSettingsPage.php:112
diff --git a/mollie-payments-for-woocommerce.php b/mollie-payments-for-woocommerce.php
index 4ff8709a..bd3dd27c 100644
--- a/mollie-payments-for-woocommerce.php
+++ b/mollie-payments-for-woocommerce.php
@@ -3,7 +3,7 @@
* Plugin Name: Mollie Payments for WooCommerce
* Plugin URI: https://www.mollie.com
* Description: Accept payments in WooCommerce with the official Mollie plugin
- * Version: 7.4.0
+ * Version: 7.4.1-beta
* Author: Mollie
* Author URI: https://www.mollie.com
* Requires at least: 5.0
@@ -19,6 +19,7 @@
namespace Mollie\WooCommerce;
+use Mollie\WooCommerce\MerchantCapture\MerchantCaptureModule;
use Mollie\WooCommerce\Vendor\Inpsyde\Modularity\Package;
use Mollie\WooCommerce\Vendor\Inpsyde\Modularity\Properties\PluginProperties;
use Mollie\WooCommerce\Activation\ActivationModule;
@@ -161,6 +162,7 @@ function initialize()
new GatewayModule(),
new VoucherModule(),
new PaymentModule(),
+ new MerchantCaptureModule(),
new UninstallModule(),
];
$modules = apply_filters('mollie_wc_plugin_modules', $modules);
diff --git a/resources/js/advancedSettings.js b/resources/js/advancedSettings.js
index d78b22aa..3aa2bc0d 100644
--- a/resources/js/advancedSettings.js
+++ b/resources/js/advancedSettings.js
@@ -1,9 +1,9 @@
(
- function ({_, jQuery }) {
+ function ({_, jQuery}) {
function mollie_settings__insertTextAtCursor(target, text, dontIgnoreSelection) {
if (target.setRangeText) {
- if ( !dontIgnoreSelection ) {
+ if (!dontIgnoreSelection) {
// insert at end
target.setRangeText(text, target.value.length, target.value.length, "end");
} else {
@@ -16,13 +16,14 @@
}
target.focus();
}
- jQuery(document).ready(function($) {
+
+ jQuery(document).ready(function ($) {
$(".mollie-settings-advanced-payment-desc-label")
.data("ignore-click", "false")
- .on("mousedown", function(e) {
+ .on("mousedown", function (e) {
const input = document.getElementById("mollie-payments-for-woocommerce_api_payment_description");
- if ( document.activeElement && input === document.activeElement ) {
- $(this).on("mouseup.molliesettings", function(e) {
+ if (document.activeElement && input === document.activeElement) {
+ $(this).on("mouseup.molliesettings", function (e) {
$(this).data("ignore-click", "true");
$(this).off(".molliesettings");
const tag = $(this).data("tag");
@@ -31,15 +32,15 @@
});
}
let $this = $(this);
- $(window).on("mouseup.molliesettings drag.molliesettings blur.molliesettings", function(e) {
+ $(window).on("mouseup.molliesettings drag.molliesettings blur.molliesettings", function (e) {
$this.off(".molliesettings");
$(window).off(".molliesettings");
});
})
- .on("click", function(e) {
+ .on("click", function (e) {
e.preventDefault();
e.stopImmediatePropagation();
- if ( $(this).data("ignore-click") === "false" ) {
+ if ($(this).data("ignore-click") === "false") {
const tag = $(this).data("tag");
const input = document.getElementById("mollie-payments-for-woocommerce_api_payment_description");
mollie_settings__insertTextAtCursor(input, tag, false);
@@ -48,9 +49,44 @@
}
})
;
+ registerManualCaptureFields();
});
}
)
(
window
)
+
+function registerManualCaptureFields() {
+ const onHoldSelect = jQuery('[name="mollie-payments-for-woocommerce_place_payment_onhold"]');
+ if (onHoldSelect.length === 0) {
+ return;
+ }
+ toggleManualCaptureFields(onHoldSelect);
+ onHoldSelect.on('change', function(){
+ toggleManualCaptureFields(onHoldSelect);
+ })
+}
+
+function toggleManualCaptureFields(onHoldSelect) {
+ const currentValue = onHoldSelect.find('option:selected');
+ if (currentValue.length === 0) {
+ return;
+ }
+
+ const captureStatusChangeField = jQuery('[name="mollie-payments-for-woocommerce_capture_or_void"]');
+ if (captureStatusChangeField.length === 0) {
+ return;
+ }
+
+ const captureStatusChangeFieldParent = captureStatusChangeField.closest('tr');
+ if (captureStatusChangeFieldParent.length === 0) {
+ return;
+ }
+
+ if (currentValue.val() === 'later_capture') {
+ captureStatusChangeFieldParent.show();
+ } else {
+ captureStatusChangeFieldParent.hide();
+ }
+}
diff --git a/src/Gateway/GatewayModule.php b/src/Gateway/GatewayModule.php
index bc5b2a18..9d997e07 100644
--- a/src/Gateway/GatewayModule.php
+++ b/src/Gateway/GatewayModule.php
@@ -6,6 +6,7 @@
namespace Mollie\WooCommerce\Gateway;
+use Automattic\WooCommerce\Internal\DataStores\Orders\CustomOrdersTableController;
use Mollie\WooCommerce\Vendor\Inpsyde\Modularity\Module\ExecutableModule;
use Mollie\WooCommerce\Vendor\Inpsyde\Modularity\Module\ModuleClassNameIdTrait;
use Mollie\WooCommerce\Vendor\Inpsyde\Modularity\Module\ServiceModule;
@@ -18,6 +19,7 @@
use Mollie\WooCommerce\Buttons\PayPalButton\PayPalButtonHandler;
use Mollie\WooCommerce\Gateway\Voucher\MaybeDisableGateway;
use Mollie\WooCommerce\Notice\AdminNotice;
+use Mollie\WooCommerce\Notice\FrontendNotice;
use Mollie\WooCommerce\Notice\NoticeInterface;
use Mollie\WooCommerce\Payment\MollieObject;
use Mollie\WooCommerce\Payment\MollieOrderService;
@@ -39,6 +41,7 @@
use Mollie\WooCommerce\PaymentMethods\Constants;
use Mollie\WooCommerce\Vendor\Psr\Container\ContainerInterface;
use Psr\Log\LoggerInterface as Logger;
+use WP_Post;
class GatewayModule implements ServiceModule, ExecutableModule
{
@@ -301,10 +304,35 @@ static function ($paymentContext) {
$order->save();
}
);
-
+ add_action('add_meta_boxes_woocommerce_page_wc-orders', [$this, 'addShopOrderMetabox'], 10);
return true;
}
+ /**
+ * @param Object $post
+ * @return void
+ */
+ public function addShopOrderMetabox(object $post)
+ {
+ if (! $post instanceof \WC_Order) {
+ return;
+ }
+ $meta = $post->get_meta('_mollie_payment_instructions');
+ if (empty($meta)) {
+ return;
+ }
+ $screen = wc_get_container()->get(CustomOrdersTableController::class)->custom_orders_table_usage_is_enabled()
+ ? wc_get_page_screen_id('shop-order')
+ : 'shop_order';
+ add_meta_box('mollie_order_details', __('Mollie Payment Details', 'mollie-payments-for-woocommerce'), static function () use ($meta) {
+ $allowedTags = ['strong' => []];
+ printf(
+ '%s
',
+ wp_kses($meta, $allowedTags)
+ );
+ }, $screen, 'side', 'high');
+ }
+
/**
* Disable Bank Transfer Gateway
*
@@ -453,8 +481,8 @@ public function instantiatePaymentMethodGateways(ContainerInterface $container):
{
$logger = $container->get(Logger::class);
assert($logger instanceof Logger);
- $notice = $container->get(AdminNotice::class);
- assert($notice instanceof AdminNotice);
+ $notice = $container->get(FrontendNotice::class);
+ assert($notice instanceof FrontendNotice);
$paymentService = $container->get(PaymentService::class);
assert($paymentService instanceof PaymentService);
$mollieOrderService = $container->get(MollieOrderService::class);
diff --git a/src/Gateway/MolliePaymentGateway.php b/src/Gateway/MolliePaymentGateway.php
index f71438a8..7047080a 100644
--- a/src/Gateway/MolliePaymentGateway.php
+++ b/src/Gateway/MolliePaymentGateway.php
@@ -30,7 +30,8 @@ class MolliePaymentGateway extends WC_Payment_Gateway implements MolliePaymentGa
/**
* @var bool
*/
- protected static $alreadyDisplayedInstructions = false;
+ protected static $alreadyDisplayedAdminInstructions = false;
+ protected static $alreadyDisplayedCustomerInstructions = false;
/**
* Recurring total, zero does not define a recurring total
*
@@ -115,7 +116,7 @@ public function __construct(
$this->paymentFactory = $paymentFactory;
$this->pluginId = $pluginId;
- // No plugin id, gateway id is unique enough
+ // No plugin id, gateway id is unique enough
$this->plugin_id = '';
// Use gateway class name as gateway id
$this->gatewayId();
@@ -156,6 +157,12 @@ public function __construct(
10,
3
);
+ add_action(
+ 'woocommerce_email_order_meta',
+ [$this, 'displayInstructions'],
+ 10,
+ 3
+ );
// Adjust title and text on Order Received page in some cases, see issue #166
add_filter('the_title', [$this, 'onOrderReceivedTitle'], 10, 2);
@@ -571,13 +578,12 @@ public function getReturnRedirectUrlForOrder(WC_Order $order): string
return $this->get_return_url($order);
} else {
$this->notice->addNotice(
- 'notice',
+ 'error',
__(
'You have cancelled your payment. Please complete your order with a different payment method.',
'mollie-payments-for-woocommerce'
)
);
-
// Return to order payment page
return $failedRedirect;
}
@@ -592,7 +598,7 @@ public function getReturnRedirectUrlForOrder(WC_Order $order): string
&& !$payment->isAuthorized()
) {
$this->notice->addNotice(
- 'notice',
+ 'error',
__(
'Your payment was not successful. Please complete your order with a different payment method.',
'mollie-payments-for-woocommerce'
@@ -606,7 +612,7 @@ public function getReturnRedirectUrlForOrder(WC_Order $order): string
}
} catch (UnexpectedValueException $exc) {
$this->notice->addNotice(
- 'notice',
+ 'error',
__(
'Your payment was not successful. Please complete your order with a different payment method.',
'mollie-payments-for-woocommerce'
@@ -807,7 +813,10 @@ public function displayInstructions(
$plain_text = false
) {
- if (!$this::$alreadyDisplayedInstructions) {
+ if (
+ ($admin_instructions && !$this::$alreadyDisplayedAdminInstructions)
+ || (!$admin_instructions && !$this::$alreadyDisplayedCustomerInstructions)
+ ) {
$order_payment_method = $order->get_payment_method();
// Invalid gateway
@@ -836,6 +845,12 @@ public function displayInstructions(
if (!empty($instructions)) {
$instructions = wptexturize($instructions);
+ //save instructions in order meta
+ $order->update_meta_data(
+ '_mollie_payment_instructions',
+ $instructions
+ );
+ $order->save();
if ($plain_text) {
echo esc_html($instructions) . PHP_EOL;
@@ -846,7 +861,12 @@ public function displayInstructions(
}
}
}
- $this::$alreadyDisplayedInstructions = true;
+ if ($admin_instructions && !$this::$alreadyDisplayedAdminInstructions) {
+ $this::$alreadyDisplayedAdminInstructions = true;
+ }
+ if (!$admin_instructions && !$this::$alreadyDisplayedCustomerInstructions) {
+ $this::$alreadyDisplayedCustomerInstructions = true;
+ }
}
/**
diff --git a/src/Gateway/Surcharge.php b/src/Gateway/Surcharge.php
index 17340bf4..8bb077a0 100644
--- a/src/Gateway/Surcharge.php
+++ b/src/Gateway/Surcharge.php
@@ -105,6 +105,10 @@ public function aboveMaxLimit(float $totalAmount, array $gatewaySettings): bool
*/
public function calculateFeeAmount(WC_Cart $cart, array $gatewaySettings): float
{
+ if (!isset($gatewaySettings['payment_surcharge'])) {
+ return 0.0;
+ }
+
$surchargeType = $gatewaySettings['payment_surcharge'];
$methodName = sprintf('calculate_%s', $surchargeType);
@@ -154,7 +158,7 @@ protected function calculate_no_fee(WC_Cart $cart, array $gatewaySettings): floa
*/
protected function calculate_fixed_fee($cart, array $gatewaySettings)
{
- return !empty($gatewaySettings[Surcharge::FIXED_FEE]) ? (float) $gatewaySettings[Surcharge::FIXED_FEE] : 0.0;
+ return !empty($gatewaySettings[Surcharge::FIXED_FEE]) ? (float)$gatewaySettings[Surcharge::FIXED_FEE] : 0.0;
}
/**
@@ -294,7 +298,12 @@ protected function name_fixed_fee_percentage($paymentMethod)
$currency = get_woocommerce_currency_symbol();
$amountPercent = $paymentMethod->getProperty(self::PERCENTAGE);
/* translators: Placeholder 1: Fee amount tag. Placeholder 2: Currency. Placeholder 3: Percentage amount. */
- return sprintf(__(' + %1$s %2$s + %3$s%% fee might apply', 'mollie-payments-for-woocommerce'), $currency, $amountFix, $amountPercent);
+ return sprintf(
+ __(' + %1$s %2$s + %3$s%% fee might apply', 'mollie-payments-for-woocommerce'),
+ $currency,
+ $amountFix,
+ $amountPercent
+ );
}
/**
diff --git a/src/MerchantCapture/Capture/Action/AbstractPaymentCaptureAction.php b/src/MerchantCapture/Capture/Action/AbstractPaymentCaptureAction.php
new file mode 100644
index 00000000..70d3ef8b
--- /dev/null
+++ b/src/MerchantCapture/Capture/Action/AbstractPaymentCaptureAction.php
@@ -0,0 +1,40 @@
+apiHelper = $apiHelper;
+ $this->settingsHelper = $settingsHelper;
+ $this->order = wc_get_order($orderId);
+ $this->logger = $logger;
+ $this->pluginId = $pluginId;
+ $this->setApiKey();
+ }
+
+ protected function setApiKey()
+ {
+ $this->apiKey = $this->settingsHelper->getApiKey();
+ }
+}
diff --git a/src/MerchantCapture/Capture/Action/CapturePayment.php b/src/MerchantCapture/Capture/Action/CapturePayment.php
new file mode 100644
index 00000000..41fd5ab7
--- /dev/null
+++ b/src/MerchantCapture/Capture/Action/CapturePayment.php
@@ -0,0 +1,65 @@
+order->get_meta('_mollie_payment_id');
+
+ if (!$paymentId) {
+ $this->logger->error('Missing Mollie payment ID in order ' . $this->order->get_id());
+ $this->order->add_order_note(
+ __(
+ 'The Mollie payment ID is missing, and we are unable to capture the funds.',
+ 'mollie-payments-for-woocommerce'
+ )
+ );
+ return;
+ }
+
+ $paymentCapturesApi = $this->apiHelper->getApiClient($this->apiKey)->paymentCaptures;
+ $captureData = [
+ 'amount' => [
+ 'currency' => $this->order->get_currency(),
+ 'value' => $this->order->get_total(),
+ ],
+ ];
+ $this->logger->debug(
+ 'SEND AN ORDER CAPTURE, orderId: ' . $this->order->get_id(
+ ) . ' transactionId: ' . $paymentId . 'Capture data: ' . json_encode($captureData)
+ );
+ $paymentCapturesApi->createForId($paymentId, $captureData);
+ $this->order->update_meta_data(
+ MerchantCaptureModule::ORDER_PAYMENT_STATUS_META_KEY,
+ ManualCaptureStatus::STATUS_WAITING
+ );
+ $this->order->add_order_note(
+ sprintf(
+ __(
+ 'The payment capture of %s has been sent successfully, and we are currently awaiting confirmation.',
+ 'mollie-payments-for-woocommerce'
+ ),
+ wc_price($this->order->get_total())
+ )
+ );
+ $this->order->save();
+ } catch (ApiException $exception) {
+ $this->logger->error($exception->getMessage());
+ $this->order->add_order_note(
+ __(
+ 'Payment Capture Failed. We encountered an issue while processing the payment capture.',
+ 'mollie-payments-for-woocommerce'
+ )
+ );
+ }
+ }
+}
diff --git a/src/MerchantCapture/Capture/Action/VoidPayment.php b/src/MerchantCapture/Capture/Action/VoidPayment.php
new file mode 100644
index 00000000..0ef1d152
--- /dev/null
+++ b/src/MerchantCapture/Capture/Action/VoidPayment.php
@@ -0,0 +1,34 @@
+order->get_meta('_mollie_payment_id');
+ $paymentCapturesApi = $this->apiHelper->getApiClient($this->apiKey)->payments;
+ try {
+ $paymentCapturesApi->cancel($paymentId);
+ $this->order->update_meta_data(
+ MerchantCaptureModule::ORDER_PAYMENT_STATUS_META_KEY,
+ ManualCaptureStatus::STATUS_VOIDED
+ );
+ $this->order->save();
+ } catch (ApiException $exception) {
+ $this->logger->error($exception->getMessage());
+ $this->order->add_order_note(
+ __(
+ 'Payment Void Failed. We encountered an issue while canceling the pre-authorized payment.',
+ 'mollie-payments-for-woocommerce'
+ )
+ );
+ }
+ }
+}
diff --git a/src/MerchantCapture/Capture/Type/ManualCapture.php b/src/MerchantCapture/Capture/Type/ManualCapture.php
new file mode 100644
index 00000000..70c890c2
--- /dev/null
+++ b/src/MerchantCapture/Capture/Type/ManualCapture.php
@@ -0,0 +1,51 @@
+container = $container;
+ add_action('woocommerce_order_actions', [$this, 'enableOrderCaptureButton'], 10, 2);
+ add_action('woocommerce_order_action_' . self::MOLLIE_MANUAL_CAPTURE_ACTION, [$this, 'manualCapture']);
+ add_filter('woocommerce_mollie_wc_gateway_creditcard_args', [$this, 'sendManualCaptureMode']);
+ }
+
+ public function enableOrderCaptureButton(array $actions, \WC_Order $order): array
+ {
+ if (!$this->container->get('merchant.manual_capture.can_capture_the_order')($order)) {
+ return $actions;
+ }
+ $actions[self::MOLLIE_MANUAL_CAPTURE_ACTION] = __(
+ 'Capture authorized Mollie payment',
+ 'mollie-payments-for-woocommerce'
+ );
+ return $actions;
+ }
+
+ public function sendManualCaptureMode(array $paymentData): array
+ {
+ if ($this->container->get('merchant.manual_capture.enabled')) {
+ $paymentData['captureMode'] = 'manual';
+ }
+ return $paymentData;
+ }
+
+ public function manualCapture(\WC_Order $order)
+ {
+
+ ($this->container->get(CapturePayment::class))($order->get_id());
+ }
+}
diff --git a/src/MerchantCapture/Capture/Type/StateChangeCapture.php b/src/MerchantCapture/Capture/Type/StateChangeCapture.php
new file mode 100644
index 00000000..9e9ec3e8
--- /dev/null
+++ b/src/MerchantCapture/Capture/Type/StateChangeCapture.php
@@ -0,0 +1,55 @@
+container = $container;
+ add_action('woocommerce_order_status_changed', [$this, "orderStatusChange"], 10, 3);
+ }
+
+ public function orderStatusChange(int $orderId, string $oldStatus, string $newStatus)
+ {
+ $stateChangeCaptureEnabled = $this->container->get('merchant.manual_capture.on_status_change_enabled');
+ if (empty($stateChangeCaptureEnabled) || $stateChangeCaptureEnabled === 'no') {
+ return;
+ }
+
+ if (!in_array($oldStatus, $this->container->get('merchant.manual_capture.void_statuses'))) {
+ return;
+ }
+
+ if (in_array($newStatus, [SharedDataDictionary::STATUS_PROCESSING, SharedDataDictionary::STATUS_COMPLETED])) {
+ $this->capturePayment($orderId);
+ return;
+ }
+
+ if ($newStatus === SharedDataDictionary::STATUS_CANCELLED) {
+ $this->voidPayment($orderId);
+ }
+ }
+
+ protected function capturePayment(int $orderId)
+ {
+ ($this->container->get(CapturePayment::class))($orderId);
+ }
+
+ protected function voidPayment(int $orderId)
+ {
+ ($this->container->get(VoidPayment::class))($orderId);
+ }
+}
diff --git a/src/MerchantCapture/ManualCaptureStatus.php b/src/MerchantCapture/ManualCaptureStatus.php
new file mode 100644
index 00000000..06c34985
--- /dev/null
+++ b/src/MerchantCapture/ManualCaptureStatus.php
@@ -0,0 +1,14 @@
+ static function () {
+ $captureType = get_option('mollie-payments-for-woocommerce_place_payment_onhold');
+ return $captureType === 'later_capture';
+ },
+ 'merchant.manual_capture.supported_methods' => static function () {
+ return ['mollie_wc_gateway_creditcard'];
+ },
+ 'merchant.manual_capture.void_statuses' => static function () {
+ return apply_filters('mollie_wc_gateway_void_order_state', [SharedDataDictionary::STATUS_ON_HOLD]);
+ },
+ 'merchant.manual_capture.capture_statuses' => static function () {
+ return apply_filters('mollie_wc_gateway_capture_order_state', [SharedDataDictionary::STATUS_ON_HOLD]);
+ },
+ 'merchant.manual_capture.is_authorized' => static function ($container) {
+ return static function (WC_Order $order) use ($container) {
+ $orderIsAuthorized = $order->get_meta(
+ self::ORDER_PAYMENT_STATUS_META_KEY
+ ) === ManualCaptureStatus::STATUS_AUTHORIZED;
+ $isManualCaptureMethod = in_array(
+ $order->get_payment_method(),
+ $container->get('merchant.manual_capture.supported_methods')
+ );
+
+ return $isManualCaptureMethod && $orderIsAuthorized;
+ };
+ },
+ 'merchant.manual_capture.is_waiting' => static function ($container) {
+ return static function (WC_Order $order) use ($container) {
+ $orderIsWaiting = $order->get_meta(
+ self::ORDER_PAYMENT_STATUS_META_KEY
+ ) === ManualCaptureStatus::STATUS_WAITING;
+ $isManualCaptureMethod = in_array(
+ $order->get_payment_method(),
+ $container->get('merchant.manual_capture.supported_methods')
+ );
+
+ return $isManualCaptureMethod && $orderIsWaiting;
+ };
+ },
+ 'merchant.manual_capture.can_capture_the_order' => static function ($container) {
+ return static function (WC_Order $order) use ($container) {
+ $orderIsAuthorized = $order->get_meta(
+ self::ORDER_PAYMENT_STATUS_META_KEY
+ ) === ManualCaptureStatus::STATUS_AUTHORIZED;
+ $isManualCaptureMethod = in_array(
+ $order->get_payment_method(),
+ $container->get('merchant.manual_capture.supported_methods')
+ );
+ $isCorrectState = in_array(
+ $order->get_status(),
+ $container->get('merchant.manual_capture.capture_statuses')
+ );
+ return $isManualCaptureMethod && $orderIsAuthorized && $isCorrectState;
+ };
+ },
+ 'merchant.manual_capture.on_status_change_enabled' => static function () {
+ return get_option('mollie-payments-for-woocommerce_capture_or_void', false);
+ },
+ CapturePayment::class => static function ($container) {
+ return static function (int $orderId) use ($container) {
+ /** @var Api $api */
+ $api = $container->get('SDK.api_helper');
+
+ /** @var Settings $settings */
+ $settings = $container->get('settings.settings_helper');
+
+ /** @var Logger $logger */
+ $logger = $container->get(Logger::class);
+
+ $pluginId = $container->get('shared.plugin_id');
+
+ return (new CapturePayment($orderId, $api, $settings, $logger, $pluginId))();
+ };
+ },
+ VoidPayment::class => static function ($container) {
+ return static function (int $orderId) use ($container) {
+ /** @var Api $api */
+ $api = $container->get('SDK.api_helper');
+
+ /** @var Settings $settings */
+ $settings = $container->get('settings.settings_helper');
+
+ /** @var Logger $logger */
+ $logger = $container->get(Logger::class);
+
+ $pluginId = $container->get('shared.plugin_id');
+
+ return (new VoidPayment($orderId, $api, $settings, $logger, $pluginId))();
+ };
+ },
+ ];
+ }
+
+ public function run(ContainerInterface $container): bool
+ {
+ $pluginId = $container->get('shared.plugin_id');
+ add_action($pluginId . '_after_webhook_action', static function (Payment $payment, WC_Order $order) use ($container) {
+ if ($payment->isAuthorized()) {
+ if (!$payment->getAmountCaptured() == 0.0) {
+ return;
+ }
+ $order->set_status(SharedDataDictionary::STATUS_ON_HOLD);
+ $order->update_meta_data(self::ORDER_PAYMENT_STATUS_META_KEY, ManualCaptureStatus::STATUS_AUTHORIZED);
+ $order->save();
+ } elseif ($payment->isPaid() && ($container->get('merchant.manual_capture.is_waiting'))($order)) {
+ $order->update_meta_data(self::ORDER_PAYMENT_STATUS_META_KEY, ManualCaptureStatus::STATUS_CAPTURED);
+ $order->save();
+ }
+ }, 10, 2);
+
+ add_action('woocommerce_order_refunded', static function (int $orderId) use ($container) {
+ $order = wc_get_order($orderId);
+ if (!is_a($order, WC_Order::class)) {
+ return;
+ }
+ $merchantCanCapture = ($container->get('merchant.manual_capture.is_authorized'))($order);
+ if ($merchantCanCapture) {
+ ($container->get(VoidPayment::class))($order->get_id());
+ }
+ });
+ add_action('woocommerce_order_actions_start', static function (int $orderId) use ($container) {
+ $order = wc_get_order($orderId);
+ if (!is_a($order, WC_Order::class)) {
+ return;
+ }
+ $paymentStatus = $order->get_meta(MerchantCaptureModule::ORDER_PAYMENT_STATUS_META_KEY, true);
+ $actionBlockParagraphs = [];
+
+ ob_start();
+ (new StatusRenderer())($paymentStatus);
+
+ $actionBlockParagraphs[] = ob_get_clean();
+ if (($container->get('merchant.manual_capture.can_capture_the_order'))($order)) {
+ $actionBlockParagraphs[] = __(
+ 'To capture the authorized payment, select capture action from the list below.',
+ 'mollie-payments-for-woocommerce'
+ );
+ } elseif (($container->get('merchant.manual_capture.is_authorized'))($order)) {
+ $actionBlockParagraphs[] = __(
+ 'Before capturing the authorized payment, ensure to set the order status to On Hold.',
+ 'mollie-payments-for-woocommerce'
+ );
+ }
+ (new OrderActionBlock())($actionBlockParagraphs);
+ });
+ add_filter(
+ 'mollie_wc_gateway_disable_ship_and_capture',
+ static function ($disableShipAndCapture, WC_Order $order) use ($container) {
+ if ($disableShipAndCapture) {
+ return true;
+ }
+ return $container->get('merchant.manual_capture.is_waiting')($order);
+ },
+ 10,
+ 2
+ );
+ new OrderListPaymentColumn();
+ new ManualCapture($container);
+ new StateChangeCapture($container);
+ return true;
+ }
+}
diff --git a/src/MerchantCapture/OrderListPaymentColumn.php b/src/MerchantCapture/OrderListPaymentColumn.php
new file mode 100644
index 00000000..6447336b
--- /dev/null
+++ b/src/MerchantCapture/OrderListPaymentColumn.php
@@ -0,0 +1,56 @@
+ $column) {
+ $newColumns[$columnId] = $column;
+ if ($columnId === 'order_number') {
+ $newColumns['mollie_capture_payment_status'] = __(
+ 'Payment Status',
+ 'mollie-payments-for-woocommerce'
+ );
+ $mollieColumnAdded = true;
+ }
+ }
+
+ if (!$mollieColumnAdded) {
+ $newColumns['mollie_capture_payment_status'] = __('Payment Status', 'mollie-payments-for-woocommerce');
+ }
+
+ return $newColumns;
+ }
+
+ public function renderColumnValue(string $column, int $orderId)
+ {
+ if ($column !== 'mollie_capture_payment_status') {
+ return;
+ }
+ /** @var \WC_Order $order */
+ $order = wc_get_order($orderId);
+
+ if (!is_a($order, \WC_Order::class)) {
+ return;
+ }
+
+ $molliePaymentStatus = $order->get_meta(MerchantCaptureModule::ORDER_PAYMENT_STATUS_META_KEY);
+
+ (new StatusRenderer())($molliePaymentStatus);
+ }
+}
diff --git a/src/MerchantCapture/UI/OrderActionBlock.php b/src/MerchantCapture/UI/OrderActionBlock.php
new file mode 100644
index 00000000..d0fbd656
--- /dev/null
+++ b/src/MerchantCapture/UI/OrderActionBlock.php
@@ -0,0 +1,19 @@
+";
+ foreach ($paragraphs as $paragraph) {
+ ?>
+ = wp_kses($paragraph, ['mark' => ['class' => []], 'span' => []]); ?>
+ ';
+ }
+}
diff --git a/src/MerchantCapture/UI/StatusButton.php b/src/MerchantCapture/UI/StatusButton.php
new file mode 100644
index 00000000..063217a5
--- /dev/null
+++ b/src/MerchantCapture/UI/StatusButton.php
@@ -0,0 +1,15 @@
+
+ = esc_html($text); ?>
+ static function (): AdminNotice {
return new AdminNotice();
},
+ FrontendNotice::class => static function (): FrontendNotice {
+ return new FrontendNotice();
+ },
];
}
}
diff --git a/src/Payment/MollieOrder.php b/src/Payment/MollieOrder.php
index 7f63813b..cb61dbc0 100644
--- a/src/Payment/MollieOrder.php
+++ b/src/Payment/MollieOrder.php
@@ -371,6 +371,13 @@ public function onWebhookCompleted(WC_Order $order, $payment, $paymentMethodTitl
if ($payment->method === 'paypal') {
$this->addPaypalTransactionIdToOrder($order);
}
+ add_filter('woocommerce_valid_order_statuses_for_payment_complete', static function ($statuses) {
+ $statuses[] = 'processing';
+ return $statuses;
+ });
+ add_filter('woocommerce_payment_complete_order_status', static function ($status) use ($order) {
+ return $order->get_status() === 'processing' ? 'completed' : $status;
+ });
$order->payment_complete($payment->id);
// Add messages to log
$this->logger->debug(__METHOD__ . ' WooCommerce payment_complete() processed and returned to ' . __METHOD__ . ' for order ' . $orderId);
diff --git a/src/Payment/MollieOrderService.php b/src/Payment/MollieOrderService.php
index 898c0686..277d7a4f 100644
--- a/src/Payment/MollieOrderService.php
+++ b/src/Payment/MollieOrderService.php
@@ -80,7 +80,7 @@ public function onWebhookAction()
$data_helper = $this->data;
$order = wc_get_order($order_id);
- if (!$order) {
+ if (!$order instanceof WC_Order) {
$this->httpResponse->setHttpResponseCode(404);
$this->logger->debug(__METHOD__ . ": Could not find order $order_id.");
return;
@@ -135,7 +135,11 @@ public function onWebhookAction()
// Log a message that webhook was called, doesn't mean the payment is actually processed
$this->logger->debug($this->gateway->id . ": Mollie payment object {$payment->id} (" . $payment->mode . ") webhook call for order {$order->get_id()}.", [true]);
+ // Get payment method title
+ $payment_method_title = $this->getPaymentMethodTitle($payment);
+ // Create the method name based on the payment status
+ $method_name = 'onWebhook' . ucfirst($payment->status);
// Order does not need a payment
if (! $this->orderNeedsPayment($order)) {
// TODO David: move to payment object?
@@ -145,7 +149,10 @@ public function onWebhookAction()
// Check and process a possible refund or chargeback
$this->processRefunds($order, $payment);
$this->processChargebacks($order, $payment);
-
+ //if the order gets updated to completed at mollie, we need to update the order status
+ if ($order->get_status() === 'processing' && $payment->isCompleted() && method_exists($payment_object, 'onWebhookCompleted')) {
+ $payment_object->onWebhookCompleted($order, $payment, $payment_method_title);
+ }
return;
}
@@ -154,12 +161,6 @@ public function onWebhookAction()
$this->setBillingAddressAfterPayment($payment, $order);
}
- // Get payment method title
- $payment_method_title = $this->getPaymentMethodTitle($payment);
-
- // Create the method name based on the payment status
- $method_name = 'onWebhook' . ucfirst($payment->status);
-
if (method_exists($payment_object, $method_name)) {
$payment_object->{$method_name}($order, $payment, $payment_method_title);
} else {
@@ -172,6 +173,7 @@ public function onWebhookAction()
));
}
+ do_action($this->pluginId . '_after_webhook_action', $payment, $order);
// Status 200
}
/**
diff --git a/src/Payment/MolliePayment.php b/src/Payment/MolliePayment.php
index 3d955c79..82636f65 100644
--- a/src/Payment/MolliePayment.php
+++ b/src/Payment/MolliePayment.php
@@ -453,6 +453,10 @@ public function refund(\WC_Order $order, $orderId, $paymentObject, $amount = nul
return new WP_Error('1', $errorMessage);
}
+ if ($paymentObject->isAuthorized()) {
+ return true;
+ }
+
if (! $paymentObject->isPaid()) {
$errorMessage = "Can not refund payment $paymentObject->id for WooCommerce order $orderId as it is not paid.";
diff --git a/src/Payment/PaymentModule.php b/src/Payment/PaymentModule.php
index 884a3d4b..d7c92734 100644
--- a/src/Payment/PaymentModule.php
+++ b/src/Payment/PaymentModule.php
@@ -167,13 +167,18 @@ public function cancelOrderOnExpiryDate()
foreach ($classNames as $gateway) {
$gatewayName = strtolower($gateway) . '_settings';
$gatewaySettings = get_option($gatewayName);
+
+ if (empty($gatewaySettings["activate_expiry_days_setting"]) || $gatewaySettings["activate_expiry_days_setting"] === 'no') {
+ continue;
+ }
+
$heldDuration = isset($gatewaySettings) && isset($gatewaySettings['order_dueDate']) ? $gatewaySettings['order_dueDate'] : 0;
if ($heldDuration < 1) {
continue;
}
$heldDurationInSeconds = $heldDuration * 60;
- if ($gateway === 'mollie_wc_gateway_bankTransfer') {
+ if ($gateway === 'Mollie_WC_Gateway_Banktransfer') {
$durationInHours = absint($heldDuration) * 24;
$durationInMinutes = $durationInHours * 60;
$heldDurationInSeconds = $durationInMinutes * 60;
@@ -339,7 +344,7 @@ public function shipAndCaptureOrderAtMollie($order_id)
// To disable automatic shipping and capturing of the Mollie order when a WooCommerce order status is updated to completed,
// store an option 'mollie-payments-for-woocommerce_disableShipOrderAtMollie' with value 1
- if (get_option($this->pluginId . '_' . 'disableShipOrderAtMollie', '0') === '1') {
+ if (apply_filters('mollie_wc_gateway_disable_ship_and_capture', get_option($this->pluginId . '_' . 'disableShipOrderAtMollie', '0') === '1', $order)) {
return;
}
@@ -474,6 +479,12 @@ public function cancelOrderAtMollie($order_id)
*/
public function handleExpiryDateCancelation($paymentMethods)
{
+ add_action(
+ 'init',
+ [$this, 'cancelOrderOnExpiryDate'],
+ 11,
+ 2
+ );
if (!$this->IsExpiryDateEnabled($paymentMethods)) {
as_unschedule_action('mollie_woocommerce_cancel_unpaid_orders');
return;
diff --git a/src/Payment/PaymentService.php b/src/Payment/PaymentService.php
index ea7b53f3..c2123d86 100644
--- a/src/Payment/PaymentService.php
+++ b/src/Payment/PaymentService.php
@@ -24,6 +24,7 @@ class PaymentService
{
public const PAYMENT_METHOD_TYPE_ORDER = 'order';
public const PAYMENT_METHOD_TYPE_PAYMENT = 'payment';
+
/**
* @var MolliePaymentGatewayI
*/
@@ -369,8 +370,7 @@ protected function processAsMollieOrder(
: '',
'orderNumber' => isset($data['orderNumber'])
? $data['orderNumber'] : '',
- 'lines' => isset($data['lines']) ? $data['lines'] : '',
- ];
+ 'lines' => isset($data['lines']) ? $data['lines'] : '', ];
$this->logger->debug(json_encode($apiCallLog));
$paymentOrder = $paymentObject;
@@ -387,6 +387,8 @@ protected function processAsMollieOrder(
}
} catch (ApiException $e) {
$this->handleMollieOutage($e);
+ //if exception is 422 do not try to create a payment
+ $this->handleMollieFraudRejection($e);
// Don't try to create a Mollie Payment for Klarna payment methods
$order_payment_method = $order->get_payment_method();
$orderMandatoryPaymentMethods = [
@@ -550,6 +552,7 @@ protected function processPaymentForMollie(
$apiKey
);
}
+
return $paymentObject;
}
@@ -719,7 +722,9 @@ protected function reportPaymentCreationFailure($orderId, $e, $paymentMethodId):
$message .= 'hii ' . $e->getMessage();
}
- $this->notice->addNotice('error', $message);
+ add_action('before_woocommerce_pay_form', static function () use ($message) {
+ wc_print_notice($message, 'error');
+ });
}
/**
@@ -873,4 +878,27 @@ public function handleMollieOutage(ApiException $e): void
);
}
}
+
+ /**
+ * Check if the exception is a fraud rejection, if so bail, log and inform user
+ * @param ApiException $e
+ * @return void
+ * @throws ApiException
+ */
+ public function handleMollieFraudRejection(ApiException $e): void
+ {
+ $isMollieFraudException = $this->apiHelper->isMollieFraudException($e);
+ if ($isMollieFraudException) {
+ $this->logger->debug(
+ "Creating payment object: The payment was declined due to suspected fraud, stopping process."
+ );
+
+ throw new ApiException(
+ __(
+ 'Payment failed due to: The payment was declined due to suspected fraud.',
+ 'mollie-payments-for-woocommerce'
+ )
+ );
+ }
+ }
}
diff --git a/src/SDK/Api.php b/src/SDK/Api.php
index 7b7cd5fb..d3526fb7 100644
--- a/src/SDK/Api.php
+++ b/src/SDK/Api.php
@@ -82,4 +82,15 @@ public function isMollieOutageException(\Mollie\Api\Exceptions\ApiException $e):
}
return false;
}
+
+ public function isMollieFraudException(\Mollie\Api\Exceptions\ApiException $e): bool
+ {
+ $isFraudCode = $e->getCode() === 422;
+ $isFraudMessage = strpos($e->getMessage(), 'The payment was declined due to suspected fraud') !== false;
+
+ if ($isFraudCode && $isFraudMessage) {
+ return true;
+ }
+ return false;
+ }
}
diff --git a/src/SDK/WordPressHttpAdapter.php b/src/SDK/WordPressHttpAdapter.php
index 1bffd185..5337cd0c 100644
--- a/src/SDK/WordPressHttpAdapter.php
+++ b/src/SDK/WordPressHttpAdapter.php
@@ -18,6 +18,7 @@ class WordPressHttpAdapter implements MollieHttpAdapterInterface
* HTTP status code for an empty ok response.
*/
const HTTP_NO_CONTENT = 204;
+ const PAYMENT_HTTP_NO_CONTENT = 202;
/**
* @param string $httpMethod
@@ -65,7 +66,7 @@ protected function parseResponse($response)
$statusCode = wp_remote_retrieve_response_code($response);
$httpBody = wp_remote_retrieve_body($response);
if (empty($httpBody)) {
- if ($statusCode === self::HTTP_NO_CONTENT) {
+ if ($statusCode === self::HTTP_NO_CONTENT || $statusCode === self::PAYMENT_HTTP_NO_CONTENT) {
return null;
}
diff --git a/src/Settings/General/MollieGeneralSettings.php b/src/Settings/General/MollieGeneralSettings.php
index b0d018b9..a97d4a38 100644
--- a/src/Settings/General/MollieGeneralSettings.php
+++ b/src/Settings/General/MollieGeneralSettings.php
@@ -258,14 +258,14 @@ public function gatewayFormFields(
'order_dueDate' => [
'title' => sprintf(__('Expiry time', 'mollie-payments-for-woocommerce')),
'type' => 'number',
+ 'custom_attributes' => ['step' => '1', 'min' => '10', 'max' => '526000'],
'description' => sprintf(
__(
- 'Number of MINUTES after the order will expire and will be canceled at Mollie and WooCommerce. A value of 0 means no expiry time will be considered.',
+ 'Number of MINUTES after the order will expire and will be canceled at Mollie and WooCommerce.',
'mollie-payments-for-woocommerce'
)
),
- 'custom_attributes' => ['step' => '1', 'min' => '0', 'max' => '526000'],
- 'default' => '0',
+ 'default' => '10',
'desc_tip' => false,
],
];
@@ -290,7 +290,7 @@ public function gatewayFormFields(
/* translators: Placeholder 1: Default order status, placeholder 2: Link to 'Hold Stock' setting */
'description' => sprintf(
__(
- 'Some payment methods take longer than a few hours to complete. The initial order state is then set to \'%1$s\'. This ensures the order is not cancelled when the setting %2$s is used.',
+ 'Some payment methods take longer than a few hours to complete. The initial order state is then set to \'%1$s\'. This ensures the order is not cancelled when the setting %2$s is used. This will also prevent the order to be canceled when expired.',
'mollie-payments-for-woocommerce'
),
wc_get_order_status_name(
diff --git a/src/Settings/Page/MollieSettingsPage.php b/src/Settings/Page/MollieSettingsPage.php
index 03f4c402..fd265b69 100644
--- a/src/Settings/Page/MollieSettingsPage.php
+++ b/src/Settings/Page/MollieSettingsPage.php
@@ -146,10 +146,10 @@ public function addGlobalSettingsFields(array $settings): array
) . '
';
$presentation = ''
- . '' . __(
+ . ''
diff --git a/src/Shared/Data.php b/src/Shared/Data.php
index 536b2866..3cb509e5 100644
--- a/src/Shared/Data.php
+++ b/src/Shared/Data.php
@@ -77,7 +77,7 @@ public function isBlockPluginActive(): bool
public function isSubscriptionPluginActive(): bool
{
- $subscriptionPlugin = is_plugin_active('woocommerce-subscriptions/woocommerce-subscriptions.php');
+ $subscriptionPlugin = class_exists('WC_Subscriptions');
return apply_filters('mollie_wc_subscription_plugin_active', $subscriptionPlugin);
}
diff --git a/tests/php/Functional/HelperMocks.php b/tests/php/Functional/HelperMocks.php
index 9ea1a44a..61a80371 100644
--- a/tests/php/Functional/HelperMocks.php
+++ b/tests/php/Functional/HelperMocks.php
@@ -7,6 +7,7 @@
use Mollie\Api\MollieApiClient;
use Mollie\WooCommerce\Gateway\MolliePaymentGateway;
use Mollie\WooCommerce\Notice\AdminNotice;
+use Mollie\WooCommerce\Payment\MollieOrder;
use Mollie\WooCommerce\Payment\MollieOrderService;
use Mollie\WooCommerce\Payment\OrderInstructionsService;
use Mollie\WooCommerce\Payment\OrderLines;
@@ -59,6 +60,12 @@ public function paymentFactory($apiClientMock){
);
}
+ public function mollieOrderMock()
+ {
+ return $this->getMockBuilder(MollieOrder::class)
+ ->disableOriginalConstructor()
+ ->getMock();
+ }
public function noticeMock()
{
return $this->getMockBuilder(AdminNotice::class)
diff --git a/tests/php/Functional/Payment/PaymentServiceTest.php b/tests/php/Functional/Payment/PaymentServiceTest.php
index b8c536b3..3ef13e93 100644
--- a/tests/php/Functional/Payment/PaymentServiceTest.php
+++ b/tests/php/Functional/Payment/PaymentServiceTest.php
@@ -3,6 +3,7 @@
namespace Mollie\WooCommerceTests\Functional\Payment;
use Mollie\Api\Endpoints\OrderEndpoint;
+use Mollie\Api\Exceptions\ApiException;
use Mollie\Api\MollieApiClient;
use Mollie\WooCommerce\Gateway\MolliePaymentGateway;
use Mollie\WooCommerce\Payment\PaymentCheckoutRedirectService;
@@ -129,6 +130,64 @@ public function processPayment_Order_success(){
self::assertEquals($expectedResult, $arrayResult);
}
+ /**
+ * @test
+ * @throws ApiException
+ */
+ public function processAsMollieOrder_BailsIf_FraudException()
+ {
+ stubs([
+ 'array_filter' => [],
+ ]);
+ $mockedException = new TestApiException();
+ $mockedException->setTestCode(422);
+ $mockedException->setTestMessage('The payment was declined due to suspected fraud');
+ $orderEndpointsMock = $this->createConfiguredMock(
+ OrderEndpoint::class,
+ [
+ 'create' => new MollieOrderResponse(),
+ ]
+ );
+ $paymentMethodId = 'Ideal';
+ $orderEndpointsMock->method('create')->with([])
+ ->will($this->throwException($mockedException));
+
+ $apiClientMock = $this->createMock(MollieApiClient::class);
+ $apiClientMock->orders = $orderEndpointsMock;
+ $voucherDefaultCategory = Voucher::NO_CATEGORY;
+
+ $testee = new PaymentService(
+ $this->helperMocks->noticeMock(),
+ $this->helperMocks->loggerMock(),
+ $this->helperMocks->paymentFactory($apiClientMock),
+ $this->helperMocks->dataHelper($apiClientMock),
+ $this->helperMocks->apiHelper($apiClientMock),
+ $this->helperMocks->settingsHelper(),
+ $this->helperMocks->pluginId(),
+ $this->paymentCheckoutService($apiClientMock),
+ $voucherDefaultCategory
+ );
+ $gateway = $this->mollieGateway($paymentMethodId, $testee);
+ $testee->setGateway($gateway);
+ $wcOrderId = 1;
+ $wcOrderKey = 'wc_order_hxZniP1zDcnM8';
+ $wcOrder = $this->wcOrder($wcOrderId, $wcOrderKey);
+ $cusomerId = 1;
+ $apiKey = 'test_test';
+ $method = new \ReflectionMethod(PaymentService::class, 'processAsMollieOrder');
+ $method->setAccessible(true);
+
+ $this->expectException(ApiException::class);
+
+ $method->invoke(
+ $testee,
+ $this->helperMocks->mollieOrderMock(),
+ $wcOrder,
+ $cusomerId,
+ $apiKey
+ );
+ }
+
protected function setUp(): void
{
@@ -451,5 +510,15 @@ public function getCheckoutUrl()
}
+class TestApiException extends \Mollie\Api\Exceptions\ApiException {
+ public function setTestCode($code) {
+ $this->code = $code;
+ }
+
+ public function setTestMessage($message) {
+ $this->message = $message;
+ }
+}
+