mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2024-11-10 13:12:50 +01:00
Merge pull request #6812 from LarsK1/eps
Stripe: EPS payment integration
This commit is contained in:
commit
26bbaa3ed0
@ -107,6 +107,7 @@ class Gateway extends StaticModel
|
|||||||
GatewayType::SEPA => ['refund' => true, 'token_billing' => true, 'webhooks' => ['source.chargeable', 'charge.succeeded']],
|
GatewayType::SEPA => ['refund' => true, 'token_billing' => true, 'webhooks' => ['source.chargeable', 'charge.succeeded']],
|
||||||
GatewayType::PRZELEWY24 => ['refund' => true, 'token_billing' => true, 'webhooks' => ['source.chargeable', 'charge.succeeded']],
|
GatewayType::PRZELEWY24 => ['refund' => true, 'token_billing' => true, 'webhooks' => ['source.chargeable', 'charge.succeeded']],
|
||||||
GatewayType::GIROPAY => ['refund' => true, 'token_billing' => true, 'webhooks' => ['source.chargeable', 'charge.succeeded']],
|
GatewayType::GIROPAY => ['refund' => true, 'token_billing' => true, 'webhooks' => ['source.chargeable', 'charge.succeeded']],
|
||||||
|
GatewayType::EPS => ['refund' => true, 'token_billing' => true, 'webhooks' => ['source.chargeable', 'charge.succeeded']],
|
||||||
GatewayType::IDEAL => ['refund' => true, 'token_billing' => true, 'webhooks' => ['source.chargeable', 'charge.succeeded']]];
|
GatewayType::IDEAL => ['refund' => true, 'token_billing' => true, 'webhooks' => ['source.chargeable', 'charge.succeeded']]];
|
||||||
|
|
||||||
case 39:
|
case 39:
|
||||||
|
@ -31,6 +31,7 @@ class GatewayType extends StaticModel
|
|||||||
const HOSTED_PAGE = 14; // For gateways that contain multiple methods.
|
const HOSTED_PAGE = 14; // For gateways that contain multiple methods.
|
||||||
const GIROPAY = 15;
|
const GIROPAY = 15;
|
||||||
const PRZELEWY24 = 16;
|
const PRZELEWY24 = 16;
|
||||||
|
const EPS = 17;
|
||||||
|
|
||||||
public function gateway()
|
public function gateway()
|
||||||
{
|
{
|
||||||
@ -75,6 +76,8 @@ class GatewayType extends StaticModel
|
|||||||
return ctrans('texts.przelewy24');
|
return ctrans('texts.przelewy24');
|
||||||
case self::GIROPAY:
|
case self::GIROPAY:
|
||||||
return ctrans('texts.giropay');
|
return ctrans('texts.giropay');
|
||||||
|
case self::EPS:
|
||||||
|
return ctrans('texts.EPS');
|
||||||
default:
|
default:
|
||||||
return 'Undefined.';
|
return 'Undefined.';
|
||||||
break;
|
break;
|
||||||
|
@ -49,7 +49,7 @@ class PaymentType extends StaticModel
|
|||||||
const HOSTED_PAGE = 38;
|
const HOSTED_PAGE = 38;
|
||||||
const GIROPAY = 39;
|
const GIROPAY = 39;
|
||||||
const PRZELEWY24 = 40;
|
const PRZELEWY24 = 40;
|
||||||
|
const EPS = 41;
|
||||||
|
|
||||||
public static function parseCardType($cardName)
|
public static function parseCardType($cardName)
|
||||||
{
|
{
|
||||||
|
141
app/PaymentDrivers/Stripe/EPS.php
Normal file
141
app/PaymentDrivers/Stripe/EPS.php
Normal file
@ -0,0 +1,141 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invoice Ninja (https://invoiceninja.com).
|
||||||
|
*
|
||||||
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
||||||
|
*
|
||||||
|
* @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com)
|
||||||
|
*
|
||||||
|
* @license https://www.elastic.co/licensing/elastic-license
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace App\PaymentDrivers\Stripe;
|
||||||
|
|
||||||
|
use App\Exceptions\PaymentFailed;
|
||||||
|
use App\Jobs\Mail\PaymentFailureMailer;
|
||||||
|
use App\Jobs\Util\SystemLogger;
|
||||||
|
use App\Models\GatewayType;
|
||||||
|
use App\Models\Payment;
|
||||||
|
use App\Models\PaymentType;
|
||||||
|
use App\Models\SystemLog;
|
||||||
|
use App\PaymentDrivers\StripePaymentDriver;
|
||||||
|
|
||||||
|
class EPS
|
||||||
|
{
|
||||||
|
/** @var StripePaymentDriver */
|
||||||
|
public StripePaymentDriver $stripe;
|
||||||
|
|
||||||
|
public function __construct(StripePaymentDriver $stripe)
|
||||||
|
{
|
||||||
|
$this->stripe = $stripe;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function authorizeView($data)
|
||||||
|
{
|
||||||
|
return render('gateways.stripe.eps.authorize', $data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function paymentView(array $data)
|
||||||
|
{
|
||||||
|
$data['gateway'] = $this->stripe;
|
||||||
|
$data['return_url'] = $this->buildReturnUrl();
|
||||||
|
$data['stripe_amount'] = $this->stripe->convertToStripeAmount($data['total']['amount_with_fee'], $this->stripe->client->currency()->precision, $this->stripe->client->currency());
|
||||||
|
$data['client'] = $this->stripe->client;
|
||||||
|
$data['customer'] = $this->stripe->findOrCreateCustomer()->id;
|
||||||
|
$data['country'] = $this->stripe->client->country->iso_3166_2;
|
||||||
|
|
||||||
|
$intent = \Stripe\PaymentIntent::create([
|
||||||
|
'amount' => $data['stripe_amount'],
|
||||||
|
'currency' => 'eur',
|
||||||
|
'payment_method_types' => ['eps'],
|
||||||
|
'customer' => $this->stripe->findOrCreateCustomer(),
|
||||||
|
'description' => $this->stripe->decodeUnicodeString(ctrans('texts.invoices') . ': ' . collect($data['invoices'])->pluck('invoice_number')),
|
||||||
|
|
||||||
|
]);
|
||||||
|
|
||||||
|
$data['pi_client_secret'] = $intent->client_secret;
|
||||||
|
|
||||||
|
$this->stripe->payment_hash->data = array_merge((array) $this->stripe->payment_hash->data, ['stripe_amount' => $data['stripe_amount']]);
|
||||||
|
$this->stripe->payment_hash->save();
|
||||||
|
|
||||||
|
return render('gateways.stripe.eps.pay', $data);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function buildReturnUrl(): string
|
||||||
|
{
|
||||||
|
return route('client.payments.response', [
|
||||||
|
'company_gateway_id' => $this->stripe->company_gateway->id,
|
||||||
|
'payment_hash' => $this->stripe->payment_hash->hash,
|
||||||
|
'payment_method_id' => GatewayType::EPS,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function paymentResponse($request)
|
||||||
|
{
|
||||||
|
$this->stripe->payment_hash->data = array_merge((array) $this->stripe->payment_hash->data, $request->all());
|
||||||
|
$this->stripe->payment_hash->save();
|
||||||
|
|
||||||
|
if ($request->redirect_status == 'succeeded') {
|
||||||
|
return $this->processSuccessfulPayment($request->payment_intent);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->processUnsuccessfulPayment();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function processSuccessfulPayment(string $payment_intent)
|
||||||
|
{
|
||||||
|
/* @todo: https://github.com/invoiceninja/invoiceninja/pull/3789/files#r436175798 */
|
||||||
|
|
||||||
|
$this->stripe->init();
|
||||||
|
|
||||||
|
$data = [
|
||||||
|
'payment_method' => $payment_intent,
|
||||||
|
'payment_type' => PaymentType::EPS,
|
||||||
|
'amount' => $this->stripe->convertFromStripeAmount($this->stripe->payment_hash->data->stripe_amount, $this->stripe->client->currency()->precision, $this->stripe->client->currency()),
|
||||||
|
'transaction_reference' => $payment_intent,
|
||||||
|
'gateway_type_id' => GatewayType::EPS,
|
||||||
|
];
|
||||||
|
|
||||||
|
$this->stripe->createPayment($data, Payment::STATUS_PENDING);
|
||||||
|
|
||||||
|
SystemLogger::dispatch(
|
||||||
|
['response' => $this->stripe->payment_hash->data, 'data' => $data],
|
||||||
|
SystemLog::CATEGORY_GATEWAY_RESPONSE,
|
||||||
|
SystemLog::EVENT_GATEWAY_SUCCESS,
|
||||||
|
SystemLog::TYPE_STRIPE,
|
||||||
|
$this->stripe->client,
|
||||||
|
$this->stripe->client->company,
|
||||||
|
);
|
||||||
|
|
||||||
|
return redirect()->route('client.payments.index');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function processUnsuccessfulPayment()
|
||||||
|
{
|
||||||
|
$server_response = $this->stripe->payment_hash->data;
|
||||||
|
|
||||||
|
PaymentFailureMailer::dispatch(
|
||||||
|
$this->stripe->client,
|
||||||
|
$server_response,
|
||||||
|
$this->stripe->client->company,
|
||||||
|
$this->stripe->convertFromStripeAmount($this->stripe->payment_hash->data->stripe_amount, $this->stripe->client->currency()->precision, $this->stripe->client->currency())
|
||||||
|
);
|
||||||
|
|
||||||
|
$message = [
|
||||||
|
'server_response' => $server_response,
|
||||||
|
'data' => $this->stripe->payment_hash->data,
|
||||||
|
];
|
||||||
|
|
||||||
|
SystemLogger::dispatch(
|
||||||
|
$message,
|
||||||
|
SystemLog::CATEGORY_GATEWAY_RESPONSE,
|
||||||
|
SystemLog::EVENT_GATEWAY_FAILURE,
|
||||||
|
SystemLog::TYPE_STRIPE,
|
||||||
|
$this->stripe->client,
|
||||||
|
$this->stripe->client->company,
|
||||||
|
);
|
||||||
|
|
||||||
|
throw new PaymentFailed('Failed to process the payment.', 500);
|
||||||
|
}
|
||||||
|
}
|
@ -36,6 +36,7 @@ use App\PaymentDrivers\Stripe\SEPA;
|
|||||||
use App\PaymentDrivers\Stripe\PRZELEWY24;
|
use App\PaymentDrivers\Stripe\PRZELEWY24;
|
||||||
use App\PaymentDrivers\Stripe\GIROPAY;
|
use App\PaymentDrivers\Stripe\GIROPAY;
|
||||||
use App\PaymentDrivers\Stripe\iDeal;
|
use App\PaymentDrivers\Stripe\iDeal;
|
||||||
|
use App\PaymentDrivers\Stripe\EPS;
|
||||||
use App\PaymentDrivers\Stripe\UpdatePaymentMethods;
|
use App\PaymentDrivers\Stripe\UpdatePaymentMethods;
|
||||||
use App\PaymentDrivers\Stripe\Utilities;
|
use App\PaymentDrivers\Stripe\Utilities;
|
||||||
use App\Utils\Traits\MakesHash;
|
use App\Utils\Traits\MakesHash;
|
||||||
@ -83,6 +84,7 @@ class StripePaymentDriver extends BaseDriver
|
|||||||
GatewayType::PRZELEWY24 => PRZELEWY24::class,
|
GatewayType::PRZELEWY24 => PRZELEWY24::class,
|
||||||
GatewayType::GIROPAY => GIROPAY::class,
|
GatewayType::GIROPAY => GIROPAY::class,
|
||||||
GatewayType::IDEAL => iDeal::class,
|
GatewayType::IDEAL => iDeal::class,
|
||||||
|
GatewayType::EPS => EPS::class,
|
||||||
];
|
];
|
||||||
|
|
||||||
const SYSTEM_LOG_TYPE = SystemLog::TYPE_STRIPE;
|
const SYSTEM_LOG_TYPE = SystemLog::TYPE_STRIPE;
|
||||||
@ -182,6 +184,13 @@ class StripePaymentDriver extends BaseDriver
|
|||||||
&& in_array($this->client->country->iso_3166_3, ["NLD"]))
|
&& in_array($this->client->country->iso_3166_3, ["NLD"]))
|
||||||
$types[] = GatewayType::IDEAL;
|
$types[] = GatewayType::IDEAL;
|
||||||
|
|
||||||
|
if ($this->client
|
||||||
|
&& $this->client->currency()
|
||||||
|
&& ($this->client->currency()->code == 'EUR')
|
||||||
|
&& isset($this->client->country)
|
||||||
|
&& in_array($this->client->country->iso_3166_3, ["AUT"]))
|
||||||
|
$types[] = GatewayType::EPS;
|
||||||
|
|
||||||
return $types;
|
return $types;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -213,6 +222,8 @@ class StripePaymentDriver extends BaseDriver
|
|||||||
break;
|
break;
|
||||||
case GatewayType::IDEAL:
|
case GatewayType::IDEAL:
|
||||||
return 'gateways.stripe.ideal';
|
return 'gateways.stripe.ideal';
|
||||||
|
case GatewayType::EPS:
|
||||||
|
return 'gateways.stripe.eps';
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
1
public/js/clients/payments/stripe-eps.js
vendored
Normal file
1
public/js/clients/payments/stripe-eps.js
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
!function(n){var r={};function o(e){if(r[e])return r[e].exports;var t=r[e]={i:e,l:!1,exports:{}};return n[e].call(t.exports,t,t.exports,o),t.l=!0,t.exports}o.m=n,o.c=r,o.d=function(e,t,n){o.o(e,t)||Object.defineProperty(e,t,{enumerable:!0,get:n})},o.r=function(e){"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},o.t=function(t,e){if(1&e&&(t=o(t)),8&e)return t;if(4&e&&"object"==typeof t&&t&&t.__esModule)return t;var n=Object.create(null);if(o.r(n),Object.defineProperty(n,"default",{enumerable:!0,value:t}),2&e&&"string"!=typeof t)for(var r in t)o.d(n,r,function(e){return t[e]}.bind(null,r));return n},o.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return o.d(t,"a",t),t},o.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},o.p="/",o(o.s=6)}({6:function(e,t,n){e.exports=n("RFiP")},RFiP:function(e,t){var n;function o(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}var r=null!==(n=null===(r=document.querySelector('meta[name="stripe-publishable-key"]'))||void 0===r?void 0:r.content)&&void 0!==n?n:"",i=null!==(n=null===(n=document.querySelector('meta[name="stripe-account-id"]'))||void 0===n?void 0:n.content)&&void 0!==n?n:"";new function t(e,n){var r=this;!function(e){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}(this),o(this,"setupStripe",function(){r.stripe=Stripe(r.key),r.stripeConnect&&(r.stripe.stripeAccount=i);let e=r.stripe.elements();return r.eps=e.create("epsBank",{style:{base:{padding:"10px 12px",color:"#32325d",fontSize:"16px","::placeholder":{color:"#aab7c4"}}}}),r.eps.mount("#eps-bank-element"),r}),o(this,"handle",function(){document.getElementById("pay-now").addEventListener("click",e=>{let t=document.getElementById("errors");if(!document.getElementById("eps-name").value)return t.textContent="Enter name",t.hidden=!1,void console.log("name");document.getElementById("pay-now").disabled=!0,document.querySelector("#pay-now > svg").classList.remove("hidden"),document.querySelector("#pay-now > span").classList.add("hidden"),this.stripe.confirmEpsPayment(document.querySelector("meta[name=pi-client-secret").content,{payment_method:{eps:r.eps,billing_details:{name:document.getElementById("eps-name").value}},return_url:document.querySelector('meta[name="return-url"]').content})})}),this.key=e,this.errors=document.getElementById("errors"),this.stripeConnect=n}(r,i).setupStripe().handle()}});
|
9
public/js/clients/payments/stripe-eps.js.LICENSE.txt
Normal file
9
public/js/clients/payments/stripe-eps.js.LICENSE.txt
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
/**
|
||||||
|
* Invoice Ninja (https://invoiceninja.com)
|
||||||
|
*
|
||||||
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
||||||
|
*
|
||||||
|
* @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com)
|
||||||
|
*
|
||||||
|
* @license https://opensource.org/licenses/AAL
|
||||||
|
*/
|
80
resources/js/clients/payments/stripe-eps.js
vendored
Normal file
80
resources/js/clients/payments/stripe-eps.js
vendored
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
/**
|
||||||
|
* Invoice Ninja (https://invoiceninja.com)
|
||||||
|
*
|
||||||
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
||||||
|
*
|
||||||
|
* @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com)
|
||||||
|
*
|
||||||
|
* @license https://opensource.org/licenses/AAL
|
||||||
|
*/
|
||||||
|
|
||||||
|
class ProcessEPSPay {
|
||||||
|
constructor(key, stripeConnect) {
|
||||||
|
this.key = key;
|
||||||
|
this.errors = document.getElementById('errors');
|
||||||
|
this.stripeConnect = stripeConnect;
|
||||||
|
}
|
||||||
|
|
||||||
|
setupStripe = () => {
|
||||||
|
this.stripe = Stripe(this.key);
|
||||||
|
|
||||||
|
if(this.stripeConnect)
|
||||||
|
this.stripe.stripeAccount = stripeConnect;
|
||||||
|
let elements = this.stripe.elements();
|
||||||
|
var options = {
|
||||||
|
style: {
|
||||||
|
base: {
|
||||||
|
padding: '10px 12px',
|
||||||
|
color: '#32325d',
|
||||||
|
fontSize: '16px',
|
||||||
|
'::placeholder': {
|
||||||
|
color: '#aab7c4'
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
this.eps = elements.create('epsBank', options);
|
||||||
|
this.eps.mount("#eps-bank-element");
|
||||||
|
return this;
|
||||||
|
};
|
||||||
|
|
||||||
|
handle = () => {
|
||||||
|
document.getElementById('pay-now').addEventListener('click', (e) => {
|
||||||
|
let errors = document.getElementById('errors');
|
||||||
|
|
||||||
|
if (!document.getElementById('eps-name').value) {
|
||||||
|
errors.textContent = "Enter name";
|
||||||
|
errors.hidden = false;
|
||||||
|
console.log("name");
|
||||||
|
return ;
|
||||||
|
}
|
||||||
|
document.getElementById('pay-now').disabled = true;
|
||||||
|
document.querySelector('#pay-now > svg').classList.remove('hidden');
|
||||||
|
document.querySelector('#pay-now > span').classList.add('hidden');
|
||||||
|
|
||||||
|
this.stripe.confirmEpsPayment(
|
||||||
|
document.querySelector('meta[name=pi-client-secret').content,
|
||||||
|
{
|
||||||
|
payment_method: {
|
||||||
|
eps: this.eps,
|
||||||
|
billing_details: {
|
||||||
|
name: document.getElementById("ideal-name").value,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
return_url: document.querySelector(
|
||||||
|
'meta[name="return-url"]'
|
||||||
|
).content,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const publishableKey = document.querySelector(
|
||||||
|
'meta[name="stripe-publishable-key"]'
|
||||||
|
)?.content ?? '';
|
||||||
|
|
||||||
|
const stripeConnect =
|
||||||
|
document.querySelector('meta[name="stripe-account-id"]')?.content ?? '';
|
||||||
|
|
||||||
|
new ProcessEPSPay(publishableKey, stripeConnect).setupStripe().handle();
|
@ -4323,7 +4323,8 @@ $LANG = array(
|
|||||||
'przelewy24' => 'Przelewy24',
|
'przelewy24' => 'Przelewy24',
|
||||||
'przelewy24_accept' => 'I declare that I have familiarized myself with the regulations and information obligation of the Przelewy24 service.'
|
'przelewy24_accept' => 'I declare that I have familiarized myself with the regulations and information obligation of the Przelewy24 service.'
|
||||||
'giropay' => 'GiroPay',
|
'giropay' => 'GiroPay',
|
||||||
'giropay_law' => 'By entering your Customer information (such as name, sort code and account number) you (the Customer) agree that this information is given voluntarily.'
|
'giropay_law' => 'By entering your Customer information (such as name, sort code and account number) you (the Customer) agree that this information is given voluntarily.',
|
||||||
|
'eps' => 'EPS',
|
||||||
);
|
);
|
||||||
|
|
||||||
return $LANG;
|
return $LANG;
|
||||||
|
@ -0,0 +1,7 @@
|
|||||||
|
@extends('portal.ninja2020.layout.payments', ['gateway_title' => ctrans('texts.bank_account'), 'card_title' => ctrans('texts.bank_account')])
|
||||||
|
|
||||||
|
@section('gateway_content')
|
||||||
|
@component('portal.ninja2020.components.general.card-element-single', ['title' => ctrans('texts.bank_account'), 'show_title' => false])
|
||||||
|
{{ __('texts.sofort_authorize_label') }}
|
||||||
|
@endcomponent
|
||||||
|
@endsection
|
@ -0,0 +1,11 @@
|
|||||||
|
<div id="stripe--payment-container">
|
||||||
|
@component('portal.ninja2020.components.general.card-element', ['title' => ctrans('texts.name')])
|
||||||
|
<label for="eps-name">
|
||||||
|
<input class="input w-full" id="eps-name" type="text" placeholder="{{ ctrans('texts.bank_account_holder') }}">
|
||||||
|
</label>
|
||||||
|
<label for="eps-bank-element"></label>
|
||||||
|
<div class="border p-4 rounded">
|
||||||
|
<div id="eps-bank-element"></div>
|
||||||
|
</div>
|
||||||
|
@endcomponent
|
||||||
|
</div>
|
@ -0,0 +1,28 @@
|
|||||||
|
@extends('portal.ninja2020.layout.payments', ['gateway_title' => 'EPS', 'card_title' => 'EPS'])
|
||||||
|
|
||||||
|
@section('gateway_head')
|
||||||
|
<meta name="stripe-publishable-key" content="{{ $gateway->getPublishableKey() }}">
|
||||||
|
<meta name="stripe-account-id" content="{{ $gateway->company_gateway->getConfigField('account_id') }}">
|
||||||
|
<meta name="return-url" content="{{ $return_url }}">
|
||||||
|
<meta name="amount" content="{{ $stripe_amount }}">
|
||||||
|
<meta name="country" content="{{ $country }}">
|
||||||
|
<meta name="customer" content="{{ $customer }}">
|
||||||
|
<meta name="pi-client-secret" content="{{ $pi_client_secret }}">
|
||||||
|
@endsection
|
||||||
|
|
||||||
|
@section('gateway_content')
|
||||||
|
<div class="alert alert-failure mb-4" hidden id="errors"></div>
|
||||||
|
|
||||||
|
@include('portal.ninja2020.gateways.includes.payment_details')
|
||||||
|
|
||||||
|
@component('portal.ninja2020.components.general.card-element', ['title' => ctrans('texts.payment_type')])
|
||||||
|
{{ ctrans('texts.eps') }} ({{ ctrans('texts.bank_transfer') }})
|
||||||
|
@endcomponent
|
||||||
|
@include('portal.ninja2020.gateways.stripe.eps.eps')
|
||||||
|
@include('portal.ninja2020.gateways.includes.pay_now')
|
||||||
|
@endsection
|
||||||
|
|
||||||
|
@push('footer')
|
||||||
|
<script src="https://js.stripe.com/v3/"></script>
|
||||||
|
<script src="{{ asset('js/clients/payments/stripe-eps.js') }}"></script>
|
||||||
|
@endpush
|
Loading…
Reference in New Issue
Block a user