Move scripts to the initially loaded page content
Hyvä Checkout uses Magewire to communicate with the server, which includes dynamic loading of components within the page.
These components can include scripts. Without strict CSP, they would be evaluated after being injected into the page.
However, evaluation is forbidden by strict CSP. Therefore, all <script>
tags must be present in the page source when initially loaded.
Multistep checkout
For multistep checkouts, even though only one step at a time is visible, all scripts used on any step must be loaded with the initial page.
Separating inline scripts from Magewire components
Having the script in a component template is convenient, but it isn't possible with strict CSP.
For example, let's take a Hyvä Checkout component and extract the JavaScript from the component template into a separate file.
The script must be loaded as part of the initial page. Magewire will update the component DOM without injecting the inline script to the page after the initial page load.
Example: Refactoring the payment method list
Original: Combined component and script
In the non-CSP version of the component, the inline script is included in the component template.
The following is an excerpt of the file before refactoring:
Hyva_Checkout::checkout/payment/method-list.phtml
?>
<div id="payment-methods">
<?php if ($methods): ?>
<script>
window.addEventListener('checkout:step:loaded', () => {
if ('<?= $escaper->escapeJs($magewire->method) ?>' && document.getElementById('payment-method-list')) {
window.dispatchEvent(new CustomEvent('checkout:payment:method-activate', { detail: { method: '<?= $escaper->escapeJs($magewire->method) ?>'} }))
}
}, { once: true })
</script>
<ol id="payment-method-list" class="space-y-4">
<?php
Step 1: Extract the script into a separate file
The code is copied into a new file method-list-activate.phtml
and registered as an Alpine constructor function using Alpine.data
.
Any method arguments must be passed using dataset attributes (the value is read with this.$el.dataset.method
- see below in step 3 how it is specified in the HTML).
Finally, the script is authorized to be executed on the page by registering it with the $hyvaCsp
view model.
Hyva_Checkout::checkout/payment/method-list-activate.phtml
/** @var Hyva\Theme\ViewModel\HyvaCsp $hyvaCsp */
?>
<script>
"use strict";
function hyvaCheckoutPaymentMethodListActivate() {
const method = this.$el.dataset.method;
window.addEventListener('checkout:step:loaded', () => {
if (method && document.getElementById('payment-method-list')) {
window.dispatchEvent(
new CustomEvent(
'checkout:payment:method-activate',
{detail: {method: method}}
)
)
}
}, { once: true })
return {}
}
window.addEventListener(
'alpine:init',
() => Alpine.data('hyvaCheckoutPaymentMethodListActivate', hyvaCheckoutPaymentMethodListActivate),
{once: true}
)
</script>
<?php $hyvaCsp->registerInlineScript() ?>
Step 2: Include the script in the initial checkout page load
The script must be available in all steps of the checkout. This can be achieved by declaring the block for the template with the extracted script using the handle hyva_checkout
.
layout/hyva_checkout.xml
<!-- ... -->
<body>
<!-- ... -->
<!-- this container loads on every checkout page in the footer -->
<referenceContainer name="magewire.plugin.scripts">
<!-- add the script -->
<block name="hyva-checkout.checkout.payment.method-list-activate"
template="Hyva_Checkout::checkout/payment/method-list-activate.phtml"
/>
</referenceContainer>
<!-- ... -->
</body>
Step 3: Use Alpine to trigger the code
The final step is activating the Alpine component on the payment-method-list
Hyva_Checkout::checkout/payment/method-list.phtml
?>
<div id="payment-methods">
<?php if ($methods): ?>
<?php /** script moved to Hyva_Checkout::checkout/payment/method-list-activate.phtml */ ?>
<ol id="payment-method-list"
x-data="hyvaCheckoutPaymentMethodListActivate"
data-method="<?= $escaper->escapeHtmlAttr($magewire->method) ?>"
class="space-y-4">
<?php
Note how the method code is passed to the JavaScript with the data-method
attribute.
Also note the $escaper
method must be changed accordingly, from escapeJs()
to escapeHtmlAttr()
.