Module: Page Builder
9 of 15 Pages
Expand the template's Page Builder functionality
An editable area is fairly flexible on its own, especially if you have multiple options for sections and lots of widgets, but let’s try to make it even more versatile.
We’ll create a view component that uses template properties to determine the number of columns the template has, and how they are laid out.
Expand the template properties
- Create a new folder, ~/Features/Shared/OptionProviders/ColumnLayout.
- Add an enumeration called
ColumnLayoutOption
, with options for different layouts with one, two, and three columns.C#ColumnLayoutOption.csusing System.ComponentModel; namespace TrainingGuides.Web.Features.Shared.OptionProviders.ColumnLayout; public enum ColumnLayoutOption { [Description("One column")] OneColumn = 0, [Description("Two columns")] TwoColumnEven = 1, [Description("Two columns - Lg/Sm")] TwoColumnLgSm = 2, [Description("Two columns - Sm/Lg")] TwoColumnSmLg = 3, [Description("Three columns")] ThreeColumnEven = 4, [Description("Three columns - Sm/Lg/Sm")] ThreeColumnSmLgSm = 5, }
- Return to the ~/Features/Products/ProductPagePageTemplateProperties file and add a new string property to represent the column layout.
- Set the property to use a dropdown component, supplied by our new enumeration.
- Add a visibility condition to ensure that it only appears if the
UsePageBuilder
property’s checkbox is true (enabled).C#ProductPagePageTemplateProperties.cs... [DropDownComponent( Label = "Column layout", ExplanationText = "Select the layout of the editable areas in the template.", DataProviderType = typeof(DropdownEnumOptionProvider<ColumnLayoutOption>), Order = 40)] [VisibleIfTrue(nameof(UsePageBuilder))] public string ColumnLayout { get; set; } = nameof(ColumnLayoutOption.OneColumn); ...
Now, if you sign in to the system, you should see the new checkbox appear in the template properties of any page using this template, but only if the checkbox property to use Page Builder is enabled. At this point, the new property won’t do anything, so let’s change that.
Add supporting entities for the view component
- If the ~/Features/Shared folder does not contain a ViewComponents folder, create it.
- Add a new class called
PageBuilderColumnViewModel
, with properties for the CSS class of the column, its identifier, and anEditableAreaOptions
object.C#PageBuilderColumnViewModel.csusing Kentico.PageBuilder.Web.Mvc; namespace TrainingGuides.Web.Features.Shared.ViewComponents; public class PageBuilderColumnViewModel { public string CssClass { get; set; } = string.Empty; public string Identifier { get; set; } = string.Empty; public EditableAreaOptions? EditableAreaOptions { get; set; } }
- Add a new ViewComponent called
PageBuilderColumnsViewComponent
. - Add a
PageBuilderColumnsViewModel
class, containing a collection ofPageBuilderColumnViewModel
objects.C#PageBuilderColumnsViewModel.csnamespace TrainingGuides.Web.Features.Shared.ViewComponents; public class PageBuilderColumnsViewModel { public IEnumerable<PageBuilderColumnViewModel> PageBuilderColumns = Enumerable.Empty<PageBuilderColumnViewModel>(); }
Implement the view component logic
Now it’s time to implement the view component. Let’s give the template a limit of three columns. It’s rare to see web pages with more than three columns, and if editors need more columns than that, they can use multi-column Page Builder sections within the template columns.
Open PageBuilderColumnsViewComponent.cs and add constants to the view component, representing the identifiers used when rendering editable areas, along with bootstrap classes for managing the sizes of the columns.
Bootstrap is already included in the training guides repository.C#PageBuilderColumnsViewComponent.cs... private const string AREA_MAIN = "areaMain"; private const string AREA_SECONDARY = "areaSecondary"; private const string AREA_TERTIARY = "areaTertiary"; private const string COL_XS = "col-md-3"; private const string COL_S = "col-md-4"; private const string COL_M = "col-md-6"; private const string COL_L = "col-md-8"; private const string COL = "col-md"; ...
Use properties to access the identifier constants.
This will allow for more complicated logic in the future. Later on in the series, we will modify this view component to work with widget zones for Page Builder sections.C#PageBuilderColumnsViewComponent.cs... private string MainIdentifier { get => AREA_MAIN; set { } } private string SecondaryIdentifier { get => AREA_SECONDARY; set { } } private string TertiaryIdentifier { get => AREA_TERTIARY; set { } } ...
Add a method that takes a
ColumnLayoutOption
parameter and uses it to determine the number of columns associated with the option.C#PageBuilderColumnsViewComponent.cs... private int GetNumberOfColumns(ColumnLayoutOption columnLayoutOption) => columnLayoutOption switch { ColumnLayoutOption.TwoColumnEven or ColumnLayoutOption.TwoColumnLgSm or ColumnLayoutOption.TwoColumnSmLg => 2, ColumnLayoutOption.ThreeColumnEven or ColumnLayoutOption.ThreeColumnSmLgSm => 3, ColumnLayoutOption.OneColumn or _ => 1 }; ...
Define a method that returns a
PageBuilderColumnViewModel
based on the index of the column, the selectedColumnLayoutOption
, and anEditableAreaOptions
object.- Assign bootstrap column classes according to the sizes dictated in the option’s name.
- For layout options where the columns are not the same width, apply the main identifier if it is the largest one.
C#PageBuilderColumnsViewComponent.cs
... private PageBuilderColumnViewModel GetColumn(int columnIndex, ColumnLayoutOption columnLayoutOption, EditableAreaOptions editableAreaOptions) { string cssClass = string.Empty; string columnIdentifier; switch (columnLayoutOption) { case ColumnLayoutOption.TwoColumnEven: //first column is main cssClass = COL_M; columnIdentifier = columnIndex == 0 ? MainIdentifier : SecondaryIdentifier; break; case ColumnLayoutOption.TwoColumnLgSm: //first column is main if (columnIndex == 0) { cssClass += COL_L; columnIdentifier = MainIdentifier; } else { cssClass += COL_S; columnIdentifier = SecondaryIdentifier; } break; case ColumnLayoutOption.TwoColumnSmLg: //second column is main if (columnIndex == 0) { cssClass += COL_S; columnIdentifier = SecondaryIdentifier; } else { cssClass += COL_L; columnIdentifier = MainIdentifier; } break; case ColumnLayoutOption.ThreeColumnEven: //middle column is main cssClass += COL_S; columnIdentifier = columnIndex == 1 ? MainIdentifier : columnIndex == 0 ? SecondaryIdentifier : TertiaryIdentifier; break; case ColumnLayoutOption.ThreeColumnSmLgSm: //middle column is main if (columnIndex == 1) { cssClass += COL_M; columnIdentifier = MainIdentifier; } else { cssClass += COL_XS; columnIdentifier = columnIndex == 0 ? SecondaryIdentifier : TertiaryIdentifier; } break; case ColumnLayoutOption.OneColumn: default: //sole column is main columnIdentifier = MainIdentifier; cssClass += COL; break; } return new PageBuilderColumnViewModel { CssClass = cssClass, Identifier = columnIdentifier, EditableAreaOptions = editableAreaOptions }; } ...
Implement the
Invoke
method withColumnLayoutOption
andEditableAreaOptions
parameters.- Use the
GetNumberOfColumns
method to construct afor
loop. In the loop, assemble a list ofPageBuilderColumn
view models using theGetColumn
method and the current index within the loop. - Use the list to create a
PageBuilderColumnsViewModel
and pass it to a view at the path ~/Features/Shared/ViewComponents/PageBuilderColumns.cshtml.ThePageBuilderColumns
view doesn’t exist yet, but don’t worry. We’ll add it in the next section.C#PageBuilderColumnsViewComponent.cs... public IViewComponentResult Invoke(ColumnLayoutOption columnLayoutOption, EditableAreaOptions editableAreaOptions) { int numberOfColumns = GetNumberOfColumns(columnLayoutOption); var columns = new List<PageBuilderColumnViewModel>(); for (int index = 0; index < numberOfColumns; index++) { columns.Add(GetColumn(index, columnLayoutOption, editableAreaOptions)); } var model = new PageBuilderColumnsViewModel { PageBuilderColumns = columns, }; return View("~/Features/Shared/ViewComponents/PageBuilderColumns.cshtml", model); } ...
- Use the
using Kentico.PageBuilder.Web.Mvc;
using Microsoft.AspNetCore.Mvc;
using TrainingGuides.Web.Features.Shared.OptionProviders.ColumnLayout;
namespace TrainingGuides.Web.Features.Shared.ViewComponents;
public class PageBuilderColumnsViewComponent : ViewComponent
{
private const string AREA_MAIN = "areaMain";
private const string AREA_SECONDARY = "areaSecondary";
private const string AREA_TERTIARY = "areaTertiary";
private const string COL_XS = "col-md-3";
private const string COL_S = "col-md-4";
private const string COL_M = "col-md-6";
private const string COL_L = "col-md-8";
private const string COL = "col-md";
private string MainIdentifier
{
get => AREA_MAIN;
set { }
}
private string SecondaryIdentifier
{
get => AREA_SECONDARY;
set { }
}
private string TertiaryIdentifier
{
get => AREA_TERTIARY;
set { }
}
public IViewComponentResult Invoke(ColumnLayoutOption columnLayoutOption,
EditableAreaOptions editableAreaOptions)
{
int numberOfColumns = GetNumberOfColumns(columnLayoutOption);
var columns = new List<PageBuilderColumnViewModel>();
for (int index = 0; index < numberOfColumns; index++)
{
columns.Add(GetColumn(index, columnLayoutOption, editableAreaOptions));
}
var model = new PageBuilderColumnsViewModel
{
PageBuilderColumns = columns,
};
return View("~/Features/Shared/ViewComponents/PageBuilderColumns.cshtml", model);
}
private int GetNumberOfColumns(ColumnLayoutOption columnLayoutOption) => columnLayoutOption switch
{
ColumnLayoutOption.TwoColumnEven or
ColumnLayoutOption.TwoColumnLgSm or
ColumnLayoutOption.TwoColumnSmLg
=> 2,
ColumnLayoutOption.ThreeColumnEven or
ColumnLayoutOption.ThreeColumnSmLgSm
=> 3,
ColumnLayoutOption.OneColumn or
_
=> 1
};
private PageBuilderColumnViewModel GetColumn(int columnIndex, ColumnLayoutOption columnLayoutOption, EditableAreaOptions editableAreaOptions)
{
string cssClass = string.Empty;
string columnIdentifier;
switch (columnLayoutOption)
{
case ColumnLayoutOption.TwoColumnEven:
//first column is main
cssClass = COL_M;
columnIdentifier = columnIndex == 0 ? MainIdentifier : SecondaryIdentifier;
break;
case ColumnLayoutOption.TwoColumnLgSm:
//first column is main
if (columnIndex == 0)
{
cssClass += COL_L;
columnIdentifier = MainIdentifier;
}
else
{
cssClass += COL_S;
columnIdentifier = SecondaryIdentifier;
}
break;
case ColumnLayoutOption.TwoColumnSmLg:
//second column is main
if (columnIndex == 0)
{
cssClass += COL_S;
columnIdentifier = SecondaryIdentifier;
}
else
{
cssClass += COL_L;
columnIdentifier = MainIdentifier;
}
break;
case ColumnLayoutOption.ThreeColumnEven:
//middle column is main
cssClass += COL_S;
columnIdentifier = columnIndex == 1
? MainIdentifier
: columnIndex == 0 ?
SecondaryIdentifier : TertiaryIdentifier;
break;
case ColumnLayoutOption.ThreeColumnSmLgSm:
//middle column is main
if (columnIndex == 1)
{
cssClass += COL_M;
columnIdentifier = MainIdentifier;
}
else
{
cssClass += COL_XS;
columnIdentifier = columnIndex == 0 ?
SecondaryIdentifier : TertiaryIdentifier;
}
break;
case ColumnLayoutOption.OneColumn:
default:
//sole column is main
columnIdentifier = MainIdentifier;
cssClass += COL;
break;
}
return new PageBuilderColumnViewModel
{
CssClass = cssClass,
Identifier = columnIdentifier,
EditableAreaOptions = editableAreaOptions
};
}
}
Create the component view
- Add a new file named PageBuilderColumns.cshtml to the ~/Features/Shared/ViewComponents folder.
- Set the
model
toPageBuilderColumnsViewModel
, and nestdiv
elements with thecontainer
androw
classes.cshtmlPageBuilderColumns.cshtml@using TrainingGuides.Web.Features.Shared.ViewComponents @model PageBuilderColumnsViewModel <div class="container tg-pb-col"> <div class="row"> </div> </div>
- Use a
foreach
loop to iterate through the model’sPageBuilderColumns
collection. - Within the loop, add a
div
styled with the column’scssClass
property.cshtmlPageBuilderColumns.cshtml... @foreach (var column in Model.PageBuilderColumns) { <div class="@column.CssClass"> </div> } ...
- Render an editable area with the
editable-area
tag helper from Xperience.- Set the
area-options
attribute to the column’sEditableAreaOptions
property.cshtmlPageBuilderColumns.cshtml... <editable-area area-identifier="@column.Identifier" area-options="column.EditableAreaOptions" /> ...
- Set the
Now, the view should look like this:
@using TrainingGuides.Web.Features.Shared.ViewComponents
@model PageBuilderColumnsViewModel
<div class="container tg-pb-col">
<div class="row">
@foreach (var column in Model.PageBuilderColumns)
{
<div class="@column.CssClass">
<editable-area area-identifier="@column.Identifier" area-options="column.EditableAreaOptions" />
</div>
}
</div>
</div>
Use the view component in the template
Now the view component is finished. Let’s add it to the template view so we can test our progress.
- Return to the ~/Features/Shared/Products/ProductPagePageTemplate.cshtml file.
- Remove the
editable-area
tag from the view, and replace it with the new view component.cshtmlProductPagePageTemplate.cshtml... @if(Model.Properties.UsePageBuilder) { <vc:page-builder-columns column-layout-option="@columnLayout" editable-area-options="@new EditableAreaOptions()" /> } ...
- In the razor code block that sets the
templateModel
variable, default the column layout to one column if the model’sColumnLayout
property can’t be parsed.cshtmlProductPagePageTemplate.cshtml@{ var templateModel = Model.GetTemplateModel<ProductPageViewModel>(); if (!Enum.TryParse(Model.Properties.ColumnLayout, out ColumnLayoutOption columnLayout)) { columnLayout = ColumnLayoutOption.OneColumn; } }
@using Microsoft.AspNetCore.Html
@using System.Globalization
@using TrainingGuides.Web.Features.Products
@using TrainingGuides.Web.Features.Products.Models
@using TrainingGuides.Web.Features.Shared.OptionProviders.ColumnLayout
@using TrainingGuides.Web.Features.Shared.OptionProviders.ColorScheme
@using TrainingGuides.Web.Features.Shared.ViewComponents
@model TemplateViewModel<ProductPagePageTemplateProperties>
@{
var templateModel = Model.GetTemplateModel<ProductPageViewModel>();
if (!Enum.TryParse(Model.Properties.ColumnLayout, out ColumnLayoutOption columnLayout))
{
columnLayout = ColumnLayoutOption.OneColumn;
}
}
<tg-component-style color-scheme="@Model.Properties.ColorScheme" corner-style="@Model.Properties.CornerStyle">
@if(Model.Properties.UsePageBuilder)
{
<vc:page-builder-columns column-layout-option="@columnLayout" editable-area-options="@new EditableAreaOptions()" />
}
else
{
<div class="tg-padding-big">
<div class="row">
<div class="col align-self-center">
<h3>@templateModel.NameHtml</h3>
<p>@templateModel.ShortDescriptionHtml</p>
</div>
<tg-styled-image src="@templateModel.Media.FirstOrDefault()?.FilePath" alt="@templateModel.Media.FirstOrDefault()?.Description" corner-style="@Model.Properties.CornerStyle" class="col c-product-img object-fit-cover" />
</div>
<div class="row">
<div class="c-pricelist">
<div class="col">
<tg-component-style color-scheme="@ColorSchemeOption.Light1" corner-style="@Model.Properties.CornerStyle" class="c-table">
@foreach (var feature in templateModel.Features)
{
<div class="c-table_row">
<div class="c-table_cell"><div class="c-table_cell">@feature.LabelHtml</div></div>
<div class="c-table_cell text-end">
<tg-product-feature-value feature="@feature"/>
</div>
</div>
}
</tg-component-style>
</div>
</div>
</div>
</div>
}
</tg-component-style>