Discount Ninja
  • Discount Ninja Developer Hub
    • Storefront API
      • Promotion Engine
        • Enable
        • JavaScript API
          • Functions
          • Events
          • Objects
      • Widgets
        • Guiding principles
          • Accessibility
          • Localization
          • Integration
          • Style
        • Announcement Bar
        • Notification
        • Offer Rules Popup
        • Product Banner
        • Promotion Summary
        • Promotion Code Field
        • Promotional Badge
    • Integration
    • Theme edits
      • App blocks
        • Product Page Banner
        • Promo Code Field
        • Promotion Summary
      • Code edits
        • Product Detail Page (PDP)
          • Price
          • Banner
          • Badge
        • Product List Page (PLP) and Collections
          • Attributes
          • Price
            • Searchanise Search & Filter
            • Globo Smart Product Filter & Search (aka SPF)
            • AI Search & Product Filter (Ultimate Search)
            • Boost AI Search & Filter (aka PFS)
          • Badge
          • Pagination
          • Quick View
        • Cart
          • Root
          • Cart item
          • Promo Code Field
          • Promotion Summary
          • Subtotal
        • Gift With Purchase
Powered by GitBook
On this page
  • Subscription apps
  • Email marketing apps
  • Loyalty apps
  • Pickup + delivery apps
  • Prefill checkout information
  • Currency conversion switchers
  • Installment apps
  • Search and filter apps
  • Page builders
  • Terms and conditions checkboxes
  • Markets
  • Other discount apps
  • Custom checkout logic
  • Show/hide custom HTML
  • Google Tag Manager Data Layer
  • Drawer cart split line items
  • Drawer cart
  • Progress bar
  • Order bump
  • Custom variant apps / swatch apps
  • Pagination on the PLP

Was this helpful?

  1. Discount Ninja Developer Hub

Integration

Discount Ninja includes out-of-the-box integration with a variety of apps and Shopify features.

PreviousPromotional BadgeNextTheme edits

Last updated 19 days ago

Was this helpful?

Subscription apps

  • The app automatically detects subscription products and can handle a checkout where one-time purchases and subscription products are combined.

  • The app was tested with:

    • Shopify's subscription app

  • Some subscription apps provide a table or box on product pages (PDP) that presents the one-time-purchase and subscription prices. If you need to display discounted prices for the one-time-purchase and subscription prices, you'll need to make some code edits. Learn more about this use case .

Email marketing apps

Discount Ninja can detect discount codes generated by the following apps:

  • Klaviyo: Email marketing

Loyalty apps

  • Discount codes generated by loyalty apps can be redeemed using Discount Ninja's promotion code field.

  • The app includes an offer type to build "wrapper" promotions to display Discount Ninja widgets when a loyalty discount code is entered in the promotion code field.

  • Discount Ninja can detect discount codes generated by the following apps:

    • Appstle Loyalty & Rewards

    • Influenci.io: Loyalty Rewards

    • Loyalty & referrals by Yotpo (Swell)

    • Loyalty, rewards and referrals by LoyaltyLion

    • Loyalty, Wishlist, Reviews UGC by Growave

    • Referral Candy

    • Smile: Rewards & Loyalty

    • Stamped.io Loyalty & Rewards

Pickup + delivery apps

  • Discount Ninja has built in integration with Zapiet.

    The integration relies on a method named Zapiet.Widget.checkoutEnabled() which must evaluate to true to allow the app to continue the checkout pipeline.

Prefill checkout information

  • Discount Ninja requires that third party developers add query parameters to the action attribute of the cart form, which is a standard technique to pass information from the cart to the checkout.

  • To prefill shipping information at checkout based on a selection made in the checkout, third party developers must ensure the action attribute of the cart form included the correct query parameters. This can be done using JavaScript.

  • Example of an action attribute, rendered by Liquid and updated by JavaScript:

<form action="/cart?checkout[shipping_address][city]=London" method="POST">
    ...
</form>

Currency conversion switchers

  • The app detects the currency based on the Shopify.currency object

  • We are compatible with all apps that rely on this object

Installment apps

  • The app can apply discounts to installment payments proposed by:

    • AfterPay

    • QuadPay

    • Sezzle

    • Hoolah

    • ShopPay

Search and filter apps

  • Rendering discounted prices in search results and collections managed by apps requires the following:

    • Discount Ninja must be able to identify where prices are rendered. This can be achieved in one of two ways:

      • Using a CSS selector

      • By marking the section with an attribute

    • For each collection product the app must, at a minimum, have access to the product handle

  • The app has been tested with the following apps:

Page builders

  • The app has different levels of support for different page builders. More information about each integration is available from the following support articles:

    • GemPages

Terms and conditions checkboxes

Behavior

The app automatically checks for known checkboxes that are related to terms and conditions. The app will not proceed the checkout pipeline until those checkboxes (if marked mandatory and visible) are checked.

Important: note that the logic to cancel the checkout based on the status of the checkbox should be present in the code of your theme (or the app that handles this checkbox). Discount Ninja does not display an error message, it relies on the theme to cancel the checkout and display a message.

Explicitly marking a checkbox

If you use a custom checkbox that isn't automatically detected by the app, you may need to mark it explicitly. To explicitly mark a checkbox as required before checkout, add the following attribute to the input field:

la-dn-checkout-required-checkbox

Syntax

<input 
    type="checkbox" 
    name="termsAndConditions" 
    form="cart" 
    la-dn-checkout-required-checkbox>
<label for="termsAndConditions">I accept the terms and conditions</label>

Markets

  • The app can be used on stores that have multiple markets configured.

  • Shopify functions mode supports most features included in Shopify Markets:

  • Caveats

    • Thresholds and fixed amounts: thresholds and fixed amount discounts are calculated based on the conversion rate defined in Shopify Markets.

Other discount apps

  • Discount Ninja strives to be as compatible as technically possible with other discount apps.

  • Note that the app is not compatible with:

    • Shopify's automatic discounts

    • Apps that use a discounted checkout prepared by the Draft Order API

    • Apps that require more than two discount functions

    • Apps that add/remove products at checkout using Checkout UI extensions

Custom checkout logic

To integrate with custom checkout logic, you can :

Show/hide custom HTML

Discount Ninja includes building blocks that can display specific HTML content to help you promote offers.

If you need to show or hide custom HTML blocks, you can use the following classes:

  • limoniapps-discountninja-whenproductdiscounted-show and limoniapps-discountninja-whenproductdiscounted-hide: these classes can be used on product pages only; they hide or show the content of the element based on whether the selected variant is discounted by Discount Ninja or not

  • limoniapps-discountninja-whenactivepromotions-show and limoniapps-discountninja-whenactivepromotions-hide: hide or show the content of the element based on whether Discount Ninja active promotions are available for the visitor or not

  • limoniapps-discountninja-whenpromotionsincart-show and limoniapps-discountninja-whenpromotionsincart-hide: hide or show the content of the element based on whether Discount Ninja promotions have been applied to the cart or not

  • limoniapps-discountninja-whencartdiscounted-show and limoniapps-discountninja-whencartdscounted-hide: hide or show the content of the element based on whether the total amount of the cart is discounted by Discount Ninja promotions or not

Google Tag Manager Data Layer

// Subscribe to the event
document.addEventListener("la:dn:product:discount:calculated", function(event) {
    // Retrieve the data about the product after discounts have been calculated
    const productVariantPriceInfo = event.detail.data[0];
    const discountedPrice = productVariantPriceInfo.discountedPrice;  
    console.log('Discounted price is ' + discountedPrice, productVariantPriceInfo);
    
    // Your code to integrate with the data layer goes here
    // ...
});

Drawer cart split line items

When your cart contains multiple items of the same product variant, Shopify will split those items into multiple rows in the cart (and drawer cart) when:

  • Line item property: One of the items has a different line item property (for example: 1 item has Color 'blue', the other item has a different value for 'Color' or does not specify a value for this property)

  • Selling plan: One of the items is sold using a selling plan (subscription) and the other is not

  • Discount: One of the items has a server-side discount associated with and the other does not

Discount Ninja applies hidden line item properties to BOGO and GWP products. This causes Shopify to split those lines.

To be able to change the quantity for each of those split lines individually, it is important that your theme updates quantity based on the key of the line item. The key of a line item is unique (also for split lines). If you update quantities based on the variant then Shopify will update the quantity of all lines that contain that variant, which leads to unexpected results.

See Shopify's documentation for more details:

Drawer cart

Discount Ninja applies discounts using Shopify Functions. Most (if not all) themes have built-in support for visualizing the discounts applied in this way. However, they require that the page is reloaded to render those discounts. This causes a broken user experience for the customer when the cart or drawer cart is refreshed using JavaScript instead of reloading the page.

To ensure the drawer cart immediately reflects the discounts, you'll need to:

Also check the following:

For the best customer experience, you'll also want to avoid that DN reloads the page when a gift is added/removed or a line is split (BOGO). To avoid reloading the page, configure DN to execute a line of JavaScript instead. This line should refresh the content of the drawer cart by fetching the content of the cart from Shopify and redrawing the drawer cart. This line is configured in the app under the Settings menu > General > Advanced > Refresh behavior

Progress bar

If your theme includes a progress bar that indicates how close a visitor is to unlocking a goal (e.g. free shipping), you may find that the progress bar isn't using the discounted total of the cart.

Discount Ninja applies discounts using Shopify Functions which ensures Shopify understands what the discounted total is. However, your theme's progress bar is not aware that discounts are applied immediately. This results in the progress bar showing incorrect data until the visitor reloads the page.

Example:

document.addEventListener("la:dn:cart:updated", function(event) {
    const discountedCart = event.detail.data[0]; 
        
    //The code below simply adds a line in the console log    
    console.log('Ready to update the progress bar based on the discounted total', discountedCart.total.discountedPrice); 

    //Run a sample implementation for a shipping bar
    updateFreeShippingBar(discountedCart.total.discountedPrice);
});

function updateFreeShippingBar(discountedTotal) {
    //In this example we assume that the threshold is $50
    //We multiply by 100 to express the threshold in cents as opposed to dollars.
    //This allows us to compare the threshold to the discounted total.
    //We then multiply by the currency conversion rate provided by Shopify,
    //this allows us to handle multiple currencies.
    const conversionRate = Shopify.currency.rate;
    const freeShippingThresholdInCents = 50 * 100 * conversionRate;    
    
    //We can now calculate the amount remaining (i.e. how much more the visitor must
    //spend to reach the threshold). Note that the discountedTotal is provided
    //in cents and in the currently select currency.
    const centsRemaining = freeShippingThresholdInCents - discountedTotal; 
    
    //Finally, we can calculate this amount as a percentage of the total.
    const percentageRemaining = Math.max(centsRemaining, 0) / freeShippingThresholdInCents * 100;
    const percentageComplete = 100 - percentageRemaining;
     
    //TODO: Update the progress bar
    //You'll need to update the code below to update the progress bar 
    //based on the discounted total.
    //How that is accomplished depends on the specific theme you use. 
    console.log(`Free shipping is ${percentageComplete}% complete, ${percentageRemaining}% left`);
}

Order bump

An order bump is an upsell that can be added to the cart with a simple checkbox. Examples include:

  • Shipping protection

  • Gift wrapping

  • Priority processing

Unfortunately order bumps are often implemented in a way that is not compatible with Discount Ninja. Specifically, the code that checks for the order bump is executed when the submit button of the cart is clicked and the code typically redirects to the checkout by setting window.location.

Custom variant apps / swatch apps

Discount Ninja automatically detects when a variant is selected on the PDP (product page). The app automatically detects this in one of the following two ways:

  • The selected variant is set in the query parameter using the variant query parameter. Example: https://my-shop.com/products/my-product?variant=123456

  • The selected variant is set as the selectedOption of the select input control with its name attribute set to id.

If your PDP is not compatible:

  • Change the code of your variant selector to allow the app to detect the change in one of the two aforementioned ways

Example of using the variant changed event:

function myVariantChangedEventHandler() {
    // This function represents the function that handles the event
    // when your custom solution or swatch app has detected a new
    // variant has been selected.
    const variantId = 123456;
    
    // Discount Ninja integration code
    document.dispatchEvent(new CustomEvent('la:dn:variant:changed'), [variantId]);
}

Pagination on the PLP

Shopify provides a number of query parameters to prefill the checkout:

(formerly known as PFS)

(SPF)

(Ultimate Search)

For information on how to configure these apps to show strikethrough pricing, please refer to .

: fixed product prices configured in Shopify Markets are used.

: exchange rates will be applied to product prices as configured in Shopify Markets.

: rounding will be applied to product prices as configured in Shopify Markets.

: discounted prices are only rounded at checkout, not in the cart.

add a checkout pipeline rule (as documented ); this is typically a good solution if you wish to execute a validation rule before proceeding to the checkout

use an event to trigger the checkout (as documented ); this is typically a good solution if you wish to take control of the checkout is some situations and leave Discount Ninja in control for other situations

Note: if your code redirects to the checkout by setting window.location you may want to leverage the above mentioned event as documented in the code snippet .

To integrate with Google's data layer you may need to leverage the discounted price calculated by the app. The app publishes this information using . The following code provides an example of how you may use the :

Enable strikethrough pricing as documented

If you wish to re-render the drawer cart after Discount Ninja has applied the discount to the cart, subscribe to this to be notified when the discount is applied.

Ensure you hide the native discount labels rendered by your theme using the limoniapps-discountninja-whenpromotionsincart-hide css class, as outlined

Ensure your theme (as opposed to the variant)

If your theme re-renders the drawer cart after it is opened, you need to instruct Discount Ninja to re-apply strikethrough pricing and re-render cart text and other widgets that may have been overwritten. Please refer to this to implement this.

To avoid this, you'll need to programmatically update the progress bar whenever discounts are applied. This can easily be achieved with the event.

This technique is not compatible with DN. Read more about how to integrate this approach properly .

Or inform the app that a new variant has been selected using a .

To handle Infinite scrolling and Load more pagination you'll need to either implement an event or mark the button. See for more details.

https://shopify.dev/docs/apps/build/checkout/cart-permalinks/create-cart-permalinks
Boost AI Search & Filter
Searchanise Search & Filter
Globo Smart Search & Product Filter
AI Search & Product Filter
this page
PageFly
Zipify Pages
Shogun
International pricing
Exchange rates
Rounding
Rounding
here
this page
here
updates the quantity of line items based on the line item key
Seal
Appstle
Recharge
Yotpo
Stay.ai
LogoCart API referenceShopify
Cart.js API /cart/change.js
here
data returned by that event
Refresh behavior JavaScript method
here
here
here
an event
event
event
Cart updated
here
variant changed event