You should be able to retrieve a subscription on the configuration page for your module in the back office (located by default at views/templates/admin/configure.tpl).

This is a simple working example that is purposefully basic. You can make the code as complex as you need to cover your specific requirements, as long as you follow the steps in this guide.

  1. Create the BillingAdapter class:

    Implement the BillingAdapter to handle API interactions with the Billing service.

    class BillingAdapter
    {
        private const BILLING_URL = 'https://billing-api.distribution.prestashop.net/v1/';
    
        private $jwt;
    
        public function __construct($jwt)
        {
            $this->jwt = $jwt;
        }
    
        public function getCurrentSubscription($shopId, $productId)
        {
            // Use Symfony client or create your own HTTP client based on curl
            $httpClient = new HttpClient(self::BILLING_URL);
            $httpClient->setHeaders([
                'Accept: application/json',
                'Authorization: Bearer ' . $this->jwt,
                'Content-Type: application/json',
                'User-Agent: module-lib-billing (' . $productId . ')',
            ]);
    
            return $httpClient->get('/customers/' . $shopId . '/subscriptions/' . $productId);
        }
    }
    
  2. Retrieve current subscription using the BillingAdapter:

    Replace the billingService with the BillingAdapter to retrieve the subscription data.

    public function getContent()
    {
        //...
    
        // Load the BillingAdapter with the JWT token
        $jwt = $psAccountsService->getOrRefreshToken();
        $billingAdapter = new BillingAdapter($jwt);
    
        // Retrieve current subscription
        $shopId = $psAccountsService->getShopUuidV4();
        $currentSubscription = $billingAdapter->getCurrentSubscription($shopId, $this->module->name);
    
        $subscription = [];
        if (!empty($currentSubscription['success'])) {
            $subscription = $currentSubscription['body'];
        }
    
        $this->context->smarty->assign([
            'subscription' => $subscription,
            'hasSubscription' => !empty($subscription),
        ]);
    
        // ...
    }
    
  3. Conditional display based on the subscription: Add the following in views/templates/admin/configure.tpl.

    <section id="billing-plan-selection" {if $hasSubscription}class="hide"{/if}>
        <!-- Display your plan and a call to action to the subscription funnel -->
    </section>
    
    <div id="ps-billing-wrapper" {if !$hasSubscription}class="hide"{/if}>
        <div id="ps-billing"></div>
        <div id="ps-billing-invoice"></div>
    </div>
    
    <div id="ps-modal"></div>
    

# Deprecated

WARNING

This document has been deprecated. While the information below remains valid for historical purposes, we recommend referring to the updated documentation for the latest guidance.

You should be able to retrieve subscription in the configuration page for your module in the back office (located by default at views/templates/admin/configure.tpl).

This is a simple working example that is purposefully basic, you can make the code as complex as you need it to be to cover your needs as long as you follow the steps in this guide.

  1. Create a JS file in your module: views/js/configure.js and inject it into <module_name>.php file.











       public function getContent()
       {
         // ...
         $this->context->smarty->assign([
           // ...
           'urlConfigureJs' => $this->getPathUri() . 'views/js/ configure.js',
         ]);
         // ...
       }
    
  2. Retrieve current subscription from Billing API and inject it into your template











































     
     
     
     
       public function getContent()
       {
         // ...
    
         /**********************
          * PrestaShop Billing *
          * *******************/
         // Billing Service
    
         // Load the service for PrestaShop Billing
         $billingService = $this->getService('<module_name>.ps_billings_service');
    
         // Retrieve plans and addons for your module
         $productComponents = $billingService->getProductComponents();
    
         $componentItems = [];
         // We test here the presence of the items property in the response's body.
         if (!empty($productComponents['body']['items']))
         {
           $componentItems = $productComponents['body']['items'];
         }
    
         // Allow the use of $componentItems in your tpl file
         $this->context->smarty->assign([
           'componentItems' => $componentItems,
         ]);
    
         // Retrieve current subscription
         $currentSubscription = $billingService->getCurrentSubscription();
    
         $subscription = [];
         // We test here the success of the request in the response's body.
         if (!empty($currentSubscription['success']))
         {
             $subscription = $currentSubscription['body'];
         }
    
         // Allow the use of $subscription & $hasSubscription in your tpl file
         $this->context->smarty->assign([
             'subscription' => $subscription,
             'hasSubscription' => !empty($subscription),
         ]);
    
         // ...
       }
    
  3. Inject views/js/configure.js as a script and add it in views/templates/admin/configure.tpl







     







    <prestashop-accounts></prestashop-accounts>
    <div id="ps-billing"></div>
    <div id="ps-modal"></div>
    
    {literal}
    <script>
       const hasSubscription = {/literal}{$hasSubscription|intval}{literal};
    </script>
    {/literal}
    
    <script src="{$urlAccountsCdn|escape:'htmlall':'UTF-8'}" rel=preload></script>
    <script src="{$urlBilling|escape:'htmlall':'UTF-8'}" rel=preload></script>
    <script src="{$urlConfigureJs|escape:'htmlall':'UTF-8'}" rel=preload></script>
    
  4. Implement views/js/configure.js to make billing work with the component instead of the initialize method

    window?.psaccountsVue?.init();
    
    let billingContext = { ...window.psBillingContext.context };
    let currentModal;
    let customer;
    
    if (window.psaccountsVue.isOnboardingCompleted() == true) {
      customer = new window.psBilling.CustomerComponent({
        context: billingContext,
        hideInvoiceList: true,
        onOpenModal,
        onEventHook,
        onOpenFunnel,
      });
      customer.render('#ps-billing');
    
      // Initialize invoice list only if we have subscription
      if (hasSubscription) {
        window.psBilling.initializeInvoiceList(billingContext, '#ps-billing-invoice');
      }
    }
    
    // Modal open / close management
    async function onCloseModal(data) {
      await Promise.all([currentModal.close(), updateCustomerProps(data)]);
    }
    
    function onOpenModal(type, data) {
      currentModal = new window.psBilling.ModalContainerComponent({
        type,
        context: {
          ...billingContext,
          ...data,
        },
        onCloseModal,
        onEventHook,
      });
      currentModal.render('#ps-modal');
    }
    
    function updateCustomerProps(data) {
      return customer.updateProps({
        context: {
          ...billingContext,
          ...data,
        },
      });
    }
    
    function showBillingInvoiceWrapper() {
      document.getElementById('ps-billing-invoice').innerHTML = '';
      window.psBilling.initializeInvoiceList(billingContext, '#ps-billing-invoice');
    }
    
    function showPlanPresenter() {
      document.getElementById('billing-plan-presenter').classList.remove('hide');
      document.getElementById('ps-billing-wrapper').classList.add('hide');
    }
    
    function showBillingWrapper() {
      document.getElementById('billing-plan-presenter').classList.add('hide');
      document.getElementById('ps-billing-wrapper').classList.remove('hide');
    }
    
    function onEventHook(type, data) {
      // Event hook listener
      switch (type) {
        case window.psBilling.EVENT_HOOK_TYPE.SUBSCRIPTION_CREATED:
          console.log('subscription created');
          showBillingWrapper();
          showBillingInvoiceWrapper();
          break;
        case window.psBilling.EVENT_HOOK_TYPE.SUBSCRIPTION_UPDATED:
          console.log('subscription updated');
          showBillingWrapper();
          showBillingInvoiceWrapper();
          break;
      }
    }
    
    function onOpenFunnel({ subscription }) {
      showPlanPresenter();
    }
    
    // Open the checkout full screen modal
    function openCheckout(pricingId) {
      const offerSelection = { offerSelection: { offerPricingId: pricingId } };
      onOpenModal(window.psBilling.MODAL_TYPE.SUBSCRIPTION_FUNNEL, offerSelection);
    }
    
  5. Display your plan in your module and hide the subscription management in views/templates/admin/configure.tpl




     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     





















       <prestashop-accounts></prestashop-accounts>
    
       <!-- You should use the billing plan library in order to display your plan -->
       <section id="billing-plan-selection" {if $hasSubscription}class="hide"{/if}>
         <h2>Select your plan</h2>
         <div id="back-button" {if !$hasSubscription}class="hide" {/if}>
           <button onclick="showBillingWrapper()"
             style="background: black;color: white; padding: 0.5rem; font-weight: bold;margin-bottom: 1.5rem;">Back to
             subscription</button>
         </div>
         <div style="width: 500px; display:flex">
           {foreach $componentItems as $item}
             <div style="border: 1px solid;padding: 2rem;text-align:center;margin-left:1rem;">
               <h3 style="font-weight: bold;margin-bottom: 1rem;">{$item['name']}</h3>
    
               <!-- Pricing information must be retrieved from billing API -->
               <p style="margin-bottom: 1rem;">{$item['price']/100}€/{$item['billingPeriodUnit']}</p>
               <!-- Pricing id must be retrieved from billing API -->
               <button onclick="openCheckout('{$item['id']}')"
                 style="background: black;color: white; padding: 0.5rem; font-weight: bold;margin-bottom: 1.5rem;">Select this
                 offer</button>
    
               {if !empty($item['features'])}
                 <div class="billing-plan__feature-group">
                   {foreach $item['features'] as $feature}
                     <div style="display: flex; flex-direction: row;">
                       <div style="display: flex; flex-direction: row; align-items: flex-start;">
                         <div class="puik-icon material-icons-round" style="font-size: 1.25rem;">check</div>
                         <div>{$feature}</div>
                       </div>
                     </div>
                   {/foreach}
                 </div>
               {/if}
           {/foreach}
         </div>
       </section>
    
       <div id="ps-billing-wrapper" {if !$hasSubscription}class="hide"{/if}>
         <div id="ps-billing"></div>
         <div id="ps-billing-invoice"></div>
       </div>
       <div id="ps-modal"></div>