Shopify’s Online Store 2.0 has transformed how themes are built and managed by using modular sections. With JSON schema, you can expose a range of settings directly in the Theme Editor, allowing store owners to customize content without having to modify the code. In this article, we’ll start by building a dynamic section for a promotion banner and then explore how to extend this method for specialized areas of your store.

Part 1: Building a Dynamic Promotion Banner Section

1. Overview

The “Dynamic Promotion Banner” is designed to be a flexible, visually appealing section that merchants can add to various parts of their store. It includes a headline (as a prominent header), subtext, background image, and a call-to-action (CTA) button. Each element is configurable via JSON schema, which makes the section fully customizable within the Theme Editor.

2. Creating the Section File

First, create a new file in your theme’s sections directory (for example, dynamic-promotion-banner.liquid). Inside, you’ll combine Liquid markup with JSON schema to define both the rendering logic and the custom settings.

Liquid Markup with a Clear Header

HTML:

<!-- dynamic-promotion-banner.liquid -->
<section class="dynamic-promotion-banner" style="background-image: url({{ section.settings.banner_image | img_url: '1600x' }});">
  <div class="banner-content" style="text-align: center; padding: 60px 20px;">
    <!-- Header Section -->
    {% if section.settings.header_title != blank %}
      <header>
        <h1 style="color: {{ section.settings.header_color }};">{{ section.settings.header_title }}</h1>
      </header>
    {% endif %}
    <!-- Promotional Headline -->
    {% if section.settings.headline != blank %}
      <h2 style="color: {{ section.settings.headline_color }};">{{ section.settings.headline }}</h2>
    {% endif %}
    <!-- Subtext -->
    {% if section.settings.subtext != blank %}
      <p style="color: {{ section.settings.subtext_color }};">{{ section.settings.subtext }}</p>
    {% endif %}
    <!-- Call-to-Action Button -->
    {% if section.settings.cta_text != blank and section.settings.cta_link != blank %}
      <a href="{{ section.settings.cta_link }}" class="btn" style="background-color: {{ section.settings.cta_bg_color }}; color: {{ section.settings.cta_text_color }};">
        {{ section.settings.cta_text }}
      </a>
    {% endif %}
  </div>
</section>

In this example, we’ve added a dedicated header section (<header>) with an <h1> tag to display a primary title. This is separate from the promotional headline and gives the section a clear structure.

3. Defining the JSON Schema

The JSON schema at the bottom of the file exposes all the configurable options in the Theme Editor. Here’s an example schema that includes settings for the header, promotional headline, subtext, background image, and CTA button.

JSON:

{% schema %}
{
  "name": "Dynamic Promotion Banner",
  "settings": [
    {
      "type": "image_picker",
      "id": "banner_image",
      "label": "Banner Background Image"
    },
    {
      "type": "text",
      "id": "header_title",
      "label": "Header Title",
      "default": "Our Special Announcement"
    },
    {
      "type": "color",
      "id": "header_color",
      "label": "Header Color",
      "default": "#333333"
    },
    {
      "type": "text",
      "id": "headline",
      "label": "Promotional Headline",
      "default": "Welcome to Our Special Offer!"
    },
    {
      "type": "color",
      "id": "headline_color",
      "label": "Headline Color",
      "default": "#ffffff"
    },
    {
      "type": "textarea",
      "id": "subtext",
      "label": "Subtext/Description",
      "default": "Enjoy our exclusive deals and offers, tailored just for you."
    },
    {
      "type": "color",
      "id": "subtext_color",
      "label": "Subtext Color",
      "default": "#ffffff"
    },
    {
      "type": "text",
      "id": "cta_text",
      "label": "CTA Button Text",
      "default": "Shop Now"
    },
    {
      "type": "url",
      "id": "cta_link",
      "label": "CTA Button Link",
      "default": "/collections/all"
    },
    {
      "type": "color",
      "id": "cta_bg_color",
      "label": "CTA Button Background Color",
      "default": "#ff0000"
    },
    {
      "type": "color",
      "id": "cta_text_color",
      "label": "CTA Button Text Color",
      "default": "#ffffff"
    }
  ],
  "presets": [
    {
      "name": "Dynamic Promotion Banner",
      "category": "Custom Sections"
    }
  ]
}
{% endschema %}

This schema gives the merchant full control over the visual aspects and content of the banner. Once saved, the section appears in the Theme Editor where the user can update settings like the header title, colors, and images interactively.

4. Integrating the Section

After creating the section file:

1. Add the Section to a Template:
For example, include it in your homepage template (index.json):

JSON:

{
  "sections": {
    "banner": {
      "type": "dynamic-promotion-banner"
    }
  },
  "order": ["banner"]
}

2. Customize via Theme Editor:
Open Online Store > Themes > Customize in your Shopify admin. You’ll see the “Dynamic Promotion Banner” listed, and you can modify all settings (including the header) in real time.

Part 2: Extending the Approach for Specific Areas

The same principles used in our promotion banner can be adapted to specialized sections, such as a custom card-section or a single product page component. The key is to adjust the JSON schema and Liquid markup to match the context and content requirements.

1. Custom Section for a Single Product Page

For a product-specific section, you might want to display additional product details, such as a unique promotional message or a custom call-to-action that appears only on the product page. Here’s how you can adjust the approach:

Liquid Markup Adaptations

  • Context Awareness: Use Shopify’s product object (product) to dynamically fetch information.
  • Conditional Rendering: Display the section only if certain conditions are met (e.g., only for products on sale).

HTML:

<!-- product-custom-promo.liquid -->
{% if product.available %}
  <section class="product-custom-promo" style="background: {{ section.settings.background_color }};">
    <div class="promo-content" style="padding: 40px;">
      {% if section.settings.product_header != blank %}
        <header>
          <h1 style="color: {{ section.settings.product_header_color }};">{{ section.settings.product_header }}</h1>
        </header>
      {% endif %}
      <p style="color: {{ section.settings.product_message_color }};">{{ section.settings.product_message }}</p>
      {% if section.settings.cta_text != blank and section.settings.cta_link != blank %}
        <a href="{{ section.settings.cta_link }}" class="btn" style="background-color: {{ section.settings.cta_bg_color }}; color: {{ section.settings.cta_text_color }};">
          {{ section.settings.cta_text }}
        </a>
      {% endif %}
    </div>
  </section>
{% endif %}


JSON Schema for a Product-Specific Section

JSON:

{% schema %}
{
  "name": "Product Custom Promotion",
  "settings": [
    {
      "type": "color",
      "id": "background_color",
      "label": "Background Color",
      "default": "#f9f9f9"
    },
    {
      "type": "text",
      "id": "product_header",
      "label": "Product Header Title",
      "default": "Special Offer for This Product"
    },
    {
      "type": "color",
      "id": "product_header_color",
      "label": "Product Header Color",
      "default": "#333333"
    },
    {
      "type": "textarea",
      "id": "product_message",
      "label": "Promotional Message",
      "default": "Get an exclusive discount when you buy this product today!"
    },
    {
      "type": "color",
      "id": "product_message_color",
      "label": "Message Text Color",
      "default": "#333333"
    },
    {
      "type": "text",
      "id": "cta_text",
      "label": "CTA Button Text",
      "default": "Buy Now"
    },
    {
      "type": "url",
      "id": "cta_link",
      "label": "CTA Button Link",
      "default": "/cart"
    },
    {
      "type": "color",
      "id": "cta_bg_color",
      "label": "CTA Button Background Color",
      "default": "#ff6600"
    },
    {
      "type": "color",
      "id": "cta_text_color",
      "label": "CTA Button Text Color",
      "default": "#ffffff"
    }
  ],
  "presets": [
    {
      "name": "Product Custom Promotion",
      "category": "Product Sections"
    }
  ]
}
{% endschema %}

In this version, the schema is tailored to a product context. The settings focus on product-related promotional messaging. You can add this section to the single product template so that it displays alongside standard product details.

2. Custom Section Within a Card-Section

If you want to include a custom card within a card-section (for instance, showcasing featured products, testimonials, or promotional content in a grid layout), you would again modify both the markup and schema.

Key Adjustments for a Card-Section:

  • Looping Through Cards: Use a loop in Liquid to iterate over multiple card items.
  • Modular Settings: Define settings for each card element (like image, title, description, etc.). You might use blocks within your JSON schema for a flexible number of cards.

Example Outline for a Card-Section

HTML:

<!-- card-section.liquid -->
<section class="custom-card-section">
  <div class="cards-wrapper">
    {% for block in section.blocks %}
      <div class="card" style="border-color: {{ block.settings.border_color }};">
        {% if block.settings.card_image != blank %}
          <img src="{{ block.settings.card_image | img_url: '500x' }}" alt="{{ block.settings.card_title }}">
        {% endif %}
        {% if block.settings.card_title != blank %}
          <h3 style="color: {{ block.settings.title_color }};">{{ block.settings.card_title }}</h3>
        {% endif %}
        <p style="color: {{ block.settings.text_color }};">{{ block.settings.card_text }}</p>
        {% if block.settings.card_link != blank %}
          <a href="{{ block.settings.card_link }}" class="btn">Learn More</a>
        {% endif %}
      </div>
    {% endfor %}
  </div>
</section>

JSON Schema with Blocks:

JSON:

{% schema %}
{
  "name": "Custom Card Section",
  "max_blocks": 5,
  "blocks": [
    {
      "type": "card",
      "name": "Card",
      "settings": [
        {
          "type": "image_picker",
          "id": "card_image",
          "label": "Card Image"
        },
        {
          "type": "text",
          "id": "card_title",
          "label": "Card Title",
          "default": "Card Title"
        },
        {
          "type": "textarea",
          "id": "card_text",
          "label": "Card Text",
          "default": "Some descriptive text for the card."
        },
        {
          "type": "url",
          "id": "card_link",
          "label": "Card Link",
          "default": "/"
        },
        {
          "type": "color",
          "id": "border_color",
          "label": "Border Color",
          "default": "#cccccc"
        },
        {
          "type": "color",
          "id": "title_color",
          "label": "Title Color",
          "default": "#333333"
        },
        {
          "type": "color",
          "id": "text_color",
          "label": "Text Color",
          "default": "#333333"
        }
      ]
    }
  ],
  "presets": [
    {
      "name": "Custom Card Section",
      "category": "Custom Sections"
    }
  ]
}
{% endschema %}

Using blocks in your JSON schema lets merchants add multiple cards dynamically through the Theme Editor. Each block corresponds to an individual card that they can customize.

About UNHYDE®

UNHYDE is a Munich-based web development agency dedicated to pushing boundaries in web development, user experience, and digital marketing. Our mission is to create high-performing digital platforms that drive meaningful customer engagement and measurable business growth. We operate internationally and are a recognized Shopify Partner agency, having successfully launched countless websites and webstores worldwide.

We're here to help !

For more insights or if you're ready to take your website to the next level, feel free to reach out to us at UNHYDE®, the web design agency. We’re always here to collaborate and craft tailored solutions that meet your unique needs.

READY. SET.

Launch

Today

get in touch

MAKE CONTACT

UNHYDE•UNHYDE•UNHYDE•UNHYDE•UNHYDE•UNHYDE•