Skip to Main content

Optimizely CMS: Content Area Item Options

An Optimizely CMS plugin that adds custom option selectors (theme, margin, padding, etc.) to content area items in the editor UI.

Editors can pick options from dropdown selectors on each content area block, and the selected values are persisted as render settings — ready for your content area renderer to apply as CSS classes or any other rendering logic.

Features

  • Define unlimited custom selectors (theme, margin, padding, …) with a simple fluent API
  • Options appear automatically in the content area item context menu
  • Restrict which options are available per block type using attributes
  • Enable selectors for all items in a specific content area using attributes on the property
  • Selected values are stored in ContentAreaItem.RenderSettings and accessible during rendering
  • Ships as a single NuGet package — no manual file copying required

Installation

dotnet add package TuyenPham.ContentAreaItemOptions

Or via the NuGet Package Manager:

Install-Package TuyenPham.ContentAreaItemOptions

Build from source:

git clone https://github.com/precise-alloy/content-area-item-options.git content-area-item-options
cd content-area-item-options
dotnet build

Setup

1. Define Your Options

Create an extension method (or add to an existing one) that builds a ContentAreaItemOptionsRegistry and calls AddContentAreaItemOptions():

using TuyenPham.ContentAreaItemOptions.DependencyInjection;
using TuyenPham.ContentAreaItemOptions.Models;

public static class ServiceCollectionExtensions
{
    public static IServiceCollection RegisterContentAreaItemOptions(
        this IServiceCollection services)
    {
        var registry = new ContentAreaItemOptionsRegistry
        {
            new ContentAreaItemOptions
            {
                AttributeName = "data-custom-theme",
                SelectorName = "theme",
                LabelPrefix = "Theme",
            }
            .Add(new ContentAreaItemOption { Id = "black", Name = "Black", CssClass = "theme-black" })
            .Add(new ContentAreaItemOption { Id = "white", Name = "White", CssClass = "theme-white" })
            .Add(new ContentAreaItemOption { Id = "blue",  Name = "Blue",  CssClass = "theme-blue" }),

            new ContentAreaItemOptions
            {
                AttributeName = "data-custom-margin",
                SelectorName = "margin",
                LabelPrefix = "Margin",
            }
            .Add(new ContentAreaItemOption { Id = "top",    Name = "Top",    CssClass = "margin-top" })
            .Add(new ContentAreaItemOption { Id = "bottom", Name = "Bottom", CssClass = "margin-bottom" })
            .Add(new ContentAreaItemOption { Id = "both",   Name = "Both",   CssClass = "margin-both" })
            .Add(new ContentAreaItemOption { Id = "none",   Name = "None",   CssClass = "margin-none" }),
        };

        services.AddContentAreaItemOptions(registry);

        return services;
    }
}

Important: AttributeName values must start with data- to be persisted in ContentAreaItem.RenderSettings by the CMS.

ContentAreaItemOptions Properties

PropertyDescription
AttributeNameThe render setting key (must start with data-). Used to store and retrieve the selected value.
SelectorNameA unique identifier for the selector. Also used as the id when fetching a single selector from the REST store.
LabelPrefixLabel shown in the editor context menu (e.g. "Theme" → displays "Theme: Blue").
DefaultLabelLabel when no option is selected. Default: "Default".
AvailabilityControls default visibility. All (default): shown for all content types. Specific: only shown for content types or content area properties with an explicit [ContentAreaItemOptions] attribute.

ContentAreaItemOption Properties

PropertyDescription
IdUnique identifier for the option (stored in render settings).
NameDisplay name shown to editors.
DescriptionOptional description/tooltip.
CssClassCSS class to apply during rendering (optional — you control how this is used).
IconClassOptional CSS class for an icon in the selector UI.

2. Register in Startup

Call your extension method in ConfigureServices:

public void ConfigureServices(IServiceCollection services)
{
    // ... other services ...

    services.RegisterContentAreaItemOptions();
}

3. Apply Options During Rendering

Override ContentAreaRenderer to read the selected values from render settings and apply them. Here's an example that collects CSS classes from all selectors:

using EPiServer.Web.Mvc.Html;
using TuyenPham.ContentAreaItemOptions.Models;

public class CustomContentAreaRenderer : ContentAreaRenderer
{
    private readonly ContentAreaItemOptionsRegistry _optionsRegistry;

    public CustomContentAreaRenderer(ContentAreaItemOptionsRegistry optionsRegistry)
    {
        _optionsRegistry = optionsRegistry;
    }

    protected override void RenderContentAreaItem(
        IHtmlHelper htmlHelper,
        ContentAreaItem contentAreaItem,
        string templateTag,
        string htmlTag,
        string cssClass)
    {
        var renderSettings = contentAreaItem.RenderSettings
            ?? new Dictionary<string, object>();

        var customClasses = GetCustomCssClasses(renderSettings);

        // Pass classes to your view via ViewBag, htmlTag, or however you render blocks
        htmlHelper.ViewBag.CustomCssClasses = customClasses;

        base.RenderContentAreaItem(htmlHelper, contentAreaItem, templateTag, htmlTag, cssClass);
    }

    private string GetCustomCssClasses(IDictionary<string, object> renderSettings)
    {
        var classes = new List<string>();

        foreach (var selector in _optionsRegistry)
        {
            if (renderSettings.TryGetValue(selector.AttributeName, out var value)
                && value is string id
                && selector.Get(id) is { CssClass: not null } option)
            {
                classes.Add(option.CssClass);
            }
        }

        return string.Join(" ", classes);
    }
}

Register the custom renderer in ConfigureServices:

services.AddTransient<ContentAreaRenderer, CustomContentAreaRenderer>();

Controlling Options with [ContentAreaItemOptions]

The [ContentAreaItemOptions] attribute can be applied to block classes (to restrict options per block type) or to ContentArea properties (to enable selectors for all items in that content area).

The behavior depends on the selector's Availability setting:

Availability = All (default)

All content types see the selector by default. Use the attribute to restrict or hide it:

using TuyenPham.ContentAreaItemOptions.Models;

// Only show "black" and "white" themes for this block
[ContentAreaItemOptions("data-custom-theme", "black", "white")]
public class HeroBlock : BlockData
{
    // ...
}

// Hide the margin selector entirely for this block (no option IDs = hidden)
[ContentAreaItemOptions("data-custom-margin")]
public class BannerBlock : BlockData
{
    // ...
}
UsageEffect
[ContentAreaItemOptions("data-custom-theme", "black", "white")]Only "black" and "white" options are shown
[ContentAreaItemOptions("data-custom-theme")]The theme selector is hidden for this block type
No attributeAll options are shown (default behavior)

Availability = Specific

The selector is hidden by default. Only content types with an explicit [ContentAreaItemOptions] attribute will see it:

var registry = new ContentAreaItemOptionsRegistry
{
    new ContentAreaItemOptions
    {
        AttributeName = "data-custom-layout",
        SelectorName = "layout",
        LabelPrefix = "Layout",
        Availability = ContentAreaItemOptionsAvailability.Specific,
    }
    .Add(new ContentAreaItemOption { Id = "wide", Name = "Wide", CssClass = "layout-wide" })
    .Add(new ContentAreaItemOption { Id = "narrow", Name = "Narrow", CssClass = "layout-narrow" }),
};
// This block opts in to the layout selector with only "wide"
[ContentAreaItemOptions("data-custom-layout", "wide")]
public class ArticleBlock : BlockData { /* ... */ }

// This block has no attribute → layout selector is hidden
public class PromoBlock : BlockData { /* ... */ }
UsageEffect
[ContentAreaItemOptions("data-custom-layout", "wide")]Only "wide" option is shown
[ContentAreaItemOptions("data-custom-layout")]The layout selector is hidden for this block type
No attributeThe layout selector is hidden (Specific mode)

The attribute can be applied multiple times on the same class, once per selector.

Enabling Options on a ContentArea Property

Instead of (or in addition to) placing the attribute on each block class, you can apply it to a ContentArea property. This enables the selector for all items placed in that content area, regardless of block type. This is especially useful with Availability = Specific.

using TuyenPham.ContentAreaItemOptions.Models;

public class StartPage : PageData
{
    // Enable the layout selector for all items in this content area (all options)
    [ContentAreaItemOptions("data-custom-layout")]
    public virtual ContentArea MainContentArea { get; set; }

    // Enable with only specific options
    [ContentAreaItemOptions("data-custom-layout", "wide")]
    public virtual ContentArea SidebarContentArea { get; set; }

    // No attribute → layout selector stays hidden (Specific mode)
    public virtual ContentArea FooterContentArea { get; set; }
}
Usage on ContentArea propertyEffect
[ContentAreaItemOptions("data-custom-layout")]All layout options are shown for items in this area
[ContentAreaItemOptions("data-custom-layout", "wide")]Only "wide" is shown for items in this area
No attributeFalls back to block-type rules / Availability setting

Precedence: Block-class attributes take priority over ContentArea property attributes. If a block type has its own [ContentAreaItemOptions] for a selector, that restriction applies even if the ContentArea property enables all options.

REST Store Endpoint

The package exposes an authorized REST store endpoint via Optimizely's [RestStore] convention:

  • GET /EPiServer/TuyenPham.ContentAreaItemOptions/Stores/content-area-options/ — Returns all selectors with their options and per-content-type restrictions
  • GET /EPiServer/TuyenPham.ContentAreaItemOptions/Stores/content-area-options/{selectorName} — Returns a single selector

The client-side initializer uses the epi.storeregistry to call this endpoint automatically — you don't need to interact with it directly. It's mentioned here for debugging purposes.

How It Works

  1. At startup, AddContentAreaItemOptions() registers the module in ProtectedModuleOptions so the CMS discovers its client-side resources and REST store
  2. When an editor opens the CMS UI, the Dojo initializer registers a content-area-options store via epi.storeregistry and fetches all selectors from the REST store endpoint
  3. For each selector, a command is added to ContentAreaEditor's context menu
  4. When the editor selects an option, the value is saved in the content area item's render settings under the AttributeName key
  5. During rendering, your ContentAreaRenderer reads the value and applies it (e.g. as a CSS class)

Requirements

  • Optimizely CMS 12 (EPiServer.CMS.Core 12.23.1+)
  • .NET 8.0+