Image 1 of 4
Image 2 of 4
Image 3 of 4
Image 4 of 4
Image 1 of 4
Image 2 of 4
Image 3 of 4
Image 4 of 4

Vendure Gift Card Plugin

Product information

By Vendure GmbH

Add support for gift cards and store credit with this full-featured plugin created by the Vendure core team.

Updates and support
Save 20%
€48.00
/ month
billed annually
Updates and support
€60.00
/ month
Support by
Support languages
en
Integration type
Paid
Category
Payment & Tax
Compatible with
^2.2.7||^3.0.0
Latest version
2.0.2
Last published
Jul 18, 2024

Vendure Gift Card Plugin

This plugin adds support for gift cards (a.k.a. gift vouchers) & store credit to your Vendure application.

Features

  • Customizable gift card designs
  • Multiple gift card configurations supported (e.g. one for digital gift cards, one for physical gift cards etc.)
  • Personalized message for gift card recipient
  • Scheduled delivery of gift cards
  • Administrators may issue gift cards / store credit to specified customers
  • Multiple gift cards may be used against a single Order
  • Gift cards may be combined with other payment methods (e.g. credit card) to pay for an Order
  • Gift cards may be partially used, the balance being kept available for future Orders.

Setup

  1. Ensure you have a TaxRate set up in your default tax zone with a rate of 0. This is required when automatically creating the Gift Card product upon bootstrap.
  2. Add the plugin to your VendureConfig:
    import { GiftCardPlugin } from '@vendure-hub/vendure-gift-card-plugin';
    
    export const config = {
      //...
      plugins: [  
        GiftCardPlugin.init({
          // The license key can be found in your Vendure Hub account
          // at https://vendure.io/account/licenses, and then clicking
          // on the install instructions for this plugin's license
          licenseKey: process.env.GIFT_CARD_PLUGIN_LICENSE_KEY,
          defaultImagePaths: [
            // The plugin ships with some generic gift card images you can use to get started 
            'path/to/node_modules/@vendure-hub/vendure-gift-card-plugin/default-gift-card-black.jpg',
            'path/to/node_modules/@vendure-hub/vendure-gift-card-plugin/default-gift-card-gold.jpg',
            'path/to/node_modules/@vendure-hub/vendure-gift-card-plugin/default-gift-card-silver.jpg',
          ],
          createDefaultGiftCardProduct: {
            // On bootstrap, a new ProductVariant will be created with the following details
            name: 'Gift Card',
            sku: 'GC01',
          },
        }),
      ],
    };
    
  3. Add the UI extensions. This plugin requires the ui-devkit in order to compile a custom Admin UI including the advanced search extensions. See our guide to extending the Admin UI.
    import { compileUiExtensions } from '@vendure/ui-devkit/compiler';
    import { GiftCardPlugin } from '@vendure-hub/vendure-gift-card-plugin';
    
    // ...
    plugins: [
      AdminUiPlugin.init({
        route: 'admin',
        port: 3002,
        app: compileUiExtensions({
          outputPath: path.join(__dirname, '../admin-ui'),
          extensions: [GiftCardPlugin.uiExtensions],
          devMode: false,
        })
      }),
    ],
    
  4. Set up delivery of gift card notifications. When a gift card is purchased, 2 emails should be sent: one to the recipient, and one confirmation of sending to the customer who bought it. If you are using Vendure’s EmailPlugin, you’ll need to set up a couple of handlers like this:
    // gift-card-email-handlers.ts
    import { EmailEventListener } from '@vendure/email-plugin';
    import { GiftCardDispatchEvent, PurchasedGiftCard } from '@vendure-hub/vendure-gift-card-plugin';
    import { ConfigService, EntityHydrator, Injector } from '@vendure/core';
    
    export const giftCardEmailHandler = new EmailEventListener('gift-card')
      .on(GiftCardDispatchEvent)
      .filter((event) => event.giftCard instanceof PurchasedGiftCard)
      .loadData((event) => loadGiftCardData(event))
      .setFrom(`{{ fromAddress }}`)
      .setRecipient((event) => {
        if (event.input?.recipient) {
          // The gift card has been manually dispatched, so use
          // the recipient email address from the input
          return event.input.recipient;
        } else {
          return (event.data.giftCard as PurchasedGiftCard).recipientEmailAddress;
        }
      })
      .setSubject(
        `You received a £{{ formatMoney giftCard.initialValue }} gift card from {{ giftCard.senderName }}!`,
      )
      .setTemplateVars((event) => ({
        giftCard: event.data.giftCard,
      }));
    
    export const giftCardDeliveryConfirmationEmailHandler = new EmailEventListener(
      'gift-card-delivery-confirmation',
    )
      .on(GiftCardDispatchEvent)
      .filter((event) => event.giftCard instanceof PurchasedGiftCard)
      .loadData((event) => loadGiftCardData(event))
      .setFrom(`{{ fromAddress }}`)
      .setRecipient(
        (event) => (event.data.giftCard as PurchasedGiftCard).purchasedWithOrder.customer!.emailAddress,
      )
      .setSubject(`Your gift card to {{ giftCard.recipientEmailAddress }} has been delivered`)
      .setTemplateVars((event) => ({
        giftCard: event.data.giftCard,
        customer: event.data.customer,
      }));
    
    async function loadGiftCardData({ event, injector }: { event: GiftCardDispatchEvent; injector: Injector }) {
      const entityHydrator = injector.get(EntityHydrator);
      const { ctx, giftCard } = event;
      await entityHydrator.hydrate(ctx, giftCard as PurchasedGiftCard, {
        relations: [
          'image',
          'purchasedWithOrder.customer',
          'purchasedWithOrder.lines.items',
          'purchasedWithOrder.shippingLines',
        ],
      });
      const { assetStorageStrategy } = injector.get(ConfigService).assetOptions;
      if (assetStorageStrategy.toAbsoluteUrl) {
        const toAbsoluteUrl = assetStorageStrategy.toAbsoluteUrl.bind(assetStorageStrategy);
        if (giftCard.image && ctx.req) {
          giftCard.image.preview = toAbsoluteUrl(ctx.req, giftCard.image.preview);
        }
      }
      return { giftCard, customer: (giftCard as PurchasedGiftCard).purchasedWithOrder.customer };
    } 
    
    import { giftCardEmailHandler, giftCardDeliveryConfirmationEmailHandler } from './gift-card-email-handlers';
    
    export const config = {
      //...
      plugins: [  
        EmailPlugin.init({
          handlers: [
            ...defaultEmailHandlers,
            giftCardEmailHandler,
            giftCardDeliveryConfirmationEmailHandler,
          ],
          // ...
        }),
      ],
    };
    
    <!-- /static/email/templates/gift-card/body.hbs -->
    {{> header title="You received a gift card!" }}
    
    <mj-section background-color="#fafafa">
      <mj-column>
        <mj-text color="#525252" align='center'>
          <h1>You received a £{{ formatMoney giftCard.initialValue }} gift card from {{ giftCard.senderName }}!</h1>
        </mj-text>
    
        <mj-image width="300px" src="{{ giftCard.image.preview }}?preset=medium" />
        <mj-text font-style='italic' font-size='18px' color="#525252">
          "{{ giftCard.message }}"
        </mj-text>
        <mj-text align='center'>
          Your code:
          <h2>{{ giftCard.code }}</h2>
        </mj-text>
      </mj-column>
    </mj-section>
    
    {{> footer }} 
    
    <!-- /static/email/templates/gift-card-delivery-confirmation/body.hbs -->
    {{> header title="Your gift card has been delivered" }}
    
    <mj-section background-color="#fafafa">
      <mj-column>
        <mj-text color="#525252" align='center'>
          <h1>Your gift card has been delivered</h1>
        </mj-text>
    
        <mj-image width="300px" src="{{ giftCard.image.preview }}?preset=medium" />
        <mj-text color="#525252">
          <p>Recipient: {{ giftCard.recipientEmailAddress }}</p>
          <p>Amount: £{{ formatMoney giftCard.initialValue }}</p>
          <p>Message: "{{ giftCard.message }}"</p>
        </mj-text>
      </mj-column>
    </mj-section>
    
    {{> footer }}
    
    If you are not using the EmailPlugin, the principle is the same - you’ll need to listen for and react to the GiftCardDispatchEvent.
  5. This plugin makes several changes to the DB schema - adding new entities and custom fields. Therefore you will need to run a migration after configuring the plugin.
  6. In the Admin UI, click the new item Sales -> Gift cards. This will take you to the gift card list. From here you can view details of existing gift cards and configure the available template images (via the drop-down menu icon in the top bar next to the “Issue Gift Card” button).

Storefront implementation

Purchasing a Gift Card

In your storefront, you should create a dedicated page for purchasing gift cards. This is because gift cards have a unique set of configuration requirements (such as recipient email address, schedule etc.) and they do not use the usual addItemToOrder mutation.

Every storefront implementation will be different, but here are the main parts you’ll need to include:

  1. Fetch the available gift card templates. Use the following query to fetch a list of templates which you can then display to the custom so they can choose the design of their gift card:
    query GetGiftCardTemplates($options: GiftCardTemplateListOptions) {
      giftCardTemplates(options: $options) {
        items {
          id
          image {
            preview
          }
        }
        totalItems
      }
    }
    
    If you have multiple gift card configurations, you can filter the templates by configName:
     {
       "options": {
          "filter": {
             "configName": {
               "contains": "digital"
             }
          }
       }
     } 
    
  2. This query will return an array of template objects, including the template ID and the image asset, which you can then display to the customer and allow them to select the desired id.
  3. The page should also have form inputs to allow the customer to configure the other aspects of the gift card:
    • value int - the value of the gift card. This will also be the price of the gift card when added to the Order.
    • templateId ID - the ID of the selected template.
    • senderName string (optional)
    • recipientEmailAddress string (optional)
    • message string - greeting to the recipient. (optional)
    • deliveryDate Date - If omitted, the gift card will be dispatched upon completion of the Order. (optional)
  4. To add to cart, execute the following mutation, passing at least a value and templateId:
    mutation AddGiftCardToOrder($input: AddGiftCardToOrderInput!) {
      addGiftCardToOrder(input: $input) {
        ...on Order {
          id
          state
          # ...etc
        }
        ...on ErrorResult {
          errorCode
          message
        }
      }
    
  5. The OrderLine type has been extended with a giftCardInput field, which allows you to display the details of the gift card in your order summary table:
    query GetActiveOrder {
      activeOrder {
        id
        # ...
        lines {
          id
          linePriceWithTax
          quantity
          giftCardInput {
            value
            senderName
            recipientEmailAddress
            message
            deliveryDate
            templateId
          }
        } 
      }
    }
    
  6. Once the Order is completed, a new GiftCard entity will be created according to the configuration entered by the customer. You’ll be able to see it in the Admin UI gift card list view.

Using a gift card on an Order

Once a gift card has been created (either via Customer purchase as described above, or via an Administrator creating one from the Admin UI), it can then be applied to an Order.

Conceptually, a gift card is actually a payment method. If you look in the Admin UI Settings -> Payment methods you should see a new gift-card-payment method has automatically been created.

  1. Since a gift card is a kind of payment method, it makes sense to allow the Customer to apply the gift card as part of the checkout flow. This is done with the following query:
    mutation ApplyGiftCardToOrder($code: String!) {
      applyGiftCardToOrder(code: $code) {
        ... on Order {
          id
          totalWithTax
          totalWithTaxAfterGiftCard
          giftCardsApplied {
            code
            balance
          }
        }
        ... on ErrorResult {
          errorCode
          message
        }
      }
    }
    
    The totalWithTaxAfterGiftCard shows how much the Customer would have to pay after deducting the available balance on any applied gift cards. You should use totalWithTaxAfterGiftCard to display the Order total everywhere you currently use totalWithTax.
  2. A gift card can be removed from the Order using this mutation:
    mutation RemoveGiftCardFromOrder($code: String!) {
      removeGiftCardFromOrder(code: $code) {
        ... on Order {
          id
          # ... etc  
        }
      }
    }
    
  3. When checking out, for each gift card in the Order.giftCardsApplied array, you should execute the regular addPaymentToOrder mutation, passing the method as 'gift-card-payment' and the metadata as { code: giftCardCode }.
  4. If the applied gift card(s) do not cover the order total, you’ll also need to add a regular payment to cover the remainder. This should be done after adding the gift card payments.

Viewing open gift card balance

A Customer can view the balance of any gift card they have partially used on previous orders by querying the new Customer.openGiftCards field from the Shop API:

query GetCustomerDetails {
  activeCustomer {
    id
    firstName
    lastName
    emailAddress
    openGiftCards {
      id
      code
      balance
    }
  }
}

Scheduling of gift card deliveries

When a gift card is purchased with the deliveryDate undefined or set to today’s date, a GiftCardDispatchEvent will be published immediately upon completion of the order. However, if the delivery date is set to some future date, the event will not get published. Instead, it will need to be manually dispatched by either:

  1. Executing the dispatchDueGiftCards mutation in the Admin API. This can be set up to run automatically each day, e.g. as part of a cron job.
  2. Manually dispatching all due gift cards via the Admin UI - when there are gift cards due, there will be a button available in the gift card list view which allows you to publish all due GiftCardDispatchEvents.
  3. Creating a CLI script which is run as a cron job to dispatch the due gift cards. See Stand-alone CLI Scripts for more information on this approach.
import { bootstrapWorker, Logger, ProductService, RequestContextService } from '@vendure/core';

import { config } from './vendure-config';

if (require.main === module) {
   dispatchDueGiftCards()
           .then(() => process.exit(0))
           .catch(err => {
              Logger.error(err);
              process.exit(1);
           });
}

async function dispatchDueGiftCards() {
   Logger.info(`Dispatching gift cards...`);
   const giftCardService = app.get(GiftCardService);
   const requestContextService = app.get(RequestContextService);
   const ctx = await requestContextService.create({
      apiType: 'admin',
   });
   await giftCardService.dispatchDueGiftCards(ctx, {
      dueDate: new Date().toISOString(),
   });
   Logger.info(`Done dispatching gift cards`);
}

Gift Card Expiry

Gift cards can be set to expire, so that after the expiry date they can no longer be used. Attemping to execute the applyGiftCardToOrder mutation on an expired gift card will return an ErrorResult.

Note: regulations regarding gift card expiry differ from country to country. For example, some countries impose a minimum duration, and some may not allow expiry at all. Check your local laws before setting up expiry.

The default duration of a gift card’s validity is set in the Gift cards > Configuration screen.

Setting the number to 0 will cause gift cards to have no expiry date.

Gift card image generation

In the Gift cards -> configuration screen, there is a toggle labeled “Generate unique image for each gift card”. When enabled, this enables a feature which will create a new Asset for each gift card, allowing the template image to be overlaid with the gift card value and code.

Assets created in this way will be tagged with the “giftcards” tag.

You can additionally specify the positioning, size, color and font of the overlaid text. To specify the available font options, you need to configure the plugin like this:

import { GiftCardPlugin } from '@vendure-hub/vendure-gift-card-plugin';

export const config = {
  //...
  plugins: [  
    GiftCardPlugin.init({
      // ...
      // Custom fonts can be specified here. The `location` should be an absolute path to the font file.  
      customFonts: [{
        fontFamily: 'Adlery Pro Swash',
        location: path.join(__dirname, '../static/fonts/Adlery-Swash-TTF.ttf'),
      }],
      availableFontStacks: [
       'Arial, Helvetica Neue, Helvetica, sans-serif',
       'Impact, Charcoal, sans-serif',
       // You can then reference any of the custom fonts by name   
       'Adlery Pro Swash',
      ],
    }),
  ],
};

Troubleshooting

Issues with Sharp

This plugin depends on the sharp image library. This is also a dependency of Vendures AssetServerPlugin which is commonly installed in Vendure projects. The version of sharp used by this plugin may be different to that used by the AssetServerPlugin. In this case, you may run into issues with errors like The specified procedure could not be found..

In this case, you can use yarn’s resolutions or npm overrides to force the sharp version to match that of the AssetServerPlugin.

Vendure
Get started

Create your first commerce experience with Vendure in less than 2 minutes

Vendure is a registered trademark. Our trademark policy ensures that our brand and products are protected. Feel free to reach out if you have any questions about our trademarks.

Newsletter

Get the latest product news and announcements delivered directly to your inbox.

© Copyright 2022 - 2024, Vendure GmbH. All rights reserved.