The official documentation that explains how to create shipping methods in Ibexa Commerce omits a crucial explanation: how to calculate the fees for the method in question?
So I'm going to fill this gap and explain how to create a fee calculator for a shipping method. I'll take as an example a simple case where we have products with free shipping:
Like many other things in Ibexa or Symfony in general, this calculator will be a tagged service.
Here is the class src/Commerce/Attribute/String/StringStorageDefinition.php :
1
2
3<?php
4declare(strict_types=1);
5namespace App\Commerce\Shipping;
6use Ibexa\Contracts\Cart\Value\CartInterface;
7use Ibexa\Contracts\ProductCatalog\CurrencyServiceInterface;
8use Ibexa\Contracts\ProductCatalog\Values\AttributeInterface;
9use Ibexa\Contracts\Shipping\ShippingMethod\CostCalculatorInterface;
10use Ibexa\Contracts\Shipping\Value\ShippingMethod\ShippingMethodInterface;
11use Ibexa\ProductCatalog\Money\DecimalMoneyFactory;
12use Money\Currency;
13use Money\Money;
14class CostCalculator implements CostCalculatorInterface
15{
16 public function __construct(
17 private readonly DecimalMoneyFactory $decimalMoneyFactory,
18 private readonly CurrencyServiceInterface $currencyService
19 ) {
20 }
21 public function calculate(ShippingMethodInterface $method, CartInterface $cart): Money
22 {
23 $isFree = true;
24 // Check for each item in the cart if it has free shipping
25 foreach ($cart->getEntries() as $entry) {
26 /** @var array<AttributeInterface> $attributes */
27 $attributes = iterator_to_array($entry->getProduct()->getAttributes());
28 if (!$attributes['free_shipping']->getValue()) {
29 $isFree = false;
30 break;
31 }
32 }
33 // Get shipping method configuration
34 $price = $method->getOptions()->get('price');
35 $currencyId = $method->getOptions()->get('currency');
36 $currency = $this->currencyService->getCurrency($currencyId);
37 // All items have free shipping, return a amount of 0
38 if ($isFree) {
39 return new Money('0', new Currency($currency->getCode()));
40 }
41 return $this->decimalMoneyFactory->getMoneyParser()->parse(
42 $price,
43 new Currency($currency->getCode())
44 );
45 }
46}
47
All that remains is to tag our service in configuration file config/services.yaml :
1
2
3 App\Commerce\Shipping\CostCalculator:
4 tags:
5 - name: ibexa.shipping.shipping.cost_calculator
6 method: custom_method # custom_method à remplacer par l'identifiant de la méthode de livraison
7
Experts Ibexa de longue date, forts de nombreux projets réussis sur Ibexa, eZ Publish et ...
La conférence annuelle Ibexa se tiendra les 30 et 31 janvier 2025 à Barcelone et ...
Data security, and in particular the security of user passwords, is an absolute priority for ...
2024 aura été une année riche en tempêtes, avec ses hauts et ses bas. Mais ...
To celebrate the release of Dataflow 5 for Symfony 7, here is some feedback on ...
🎯 Nous relevons le défi de lancer un grand concours : Vous faire gagner un ...
How to dynamically calculate shipping costs? The official documentation that explains how to create shipping ...
How to create a new product attribute type in Ibexa Commerce
A la recherche d'un poste de travail temporaire ou permanent ? Vous recherchez un environnement ...
Experts Ibexa de longue date, forts de nombreux projets réussis sur Ibexa, eZ Publish et ...
La conférence annuelle Ibexa se tiendra les 30 et 31 janvier 2025 à Barcelone et ...
Data security, and in particular the security of user passwords, is an absolute priority for ...
2024 aura été une année riche en tempêtes, avec ses hauts et ses bas. Mais ...
To celebrate the release of Dataflow 5 for Symfony 7, here is some feedback on ...
🎯 Nous relevons le défi de lancer un grand concours : Vous faire gagner un ...
How to dynamically calculate shipping costs? The official documentation that explains how to create shipping ...
How to create a new product attribute type in Ibexa Commerce
A la recherche d'un poste de travail temporaire ou permanent ? Vous recherchez un environnement ...