In this blog post, I will tell you how we built a custom Sitecore JSS Snippet component in a project based on Sitecore 9.2 SXA + JSS and explain how it helped solve a client’s requirement.
The Problem
Our client needed a possibility to create reusable macrocomponents from a bunch of microcomponents and use them across pages and sites included in one JSS tenant.
Let’s assume we have the following list of JSS components (microcomponents):
- Section — a root component
- Grid Component
- Rich Text Component
- Image Component.
And we want to build a macrocomponet using these microcomponents as follows:
- Put a Grid with two columns in a Section placeholder.
- Put an Image Component in the Left Column.
- Put a Rich Text Component in the Right Column.
data:image/s3,"s3://crabby-images/b7687/b768795288e258d27e3fe0e5af742533f6ba83fa" alt="Macrocomponent Sample"
data:image/s3,"s3://crabby-images/7c4b9/7c4b9780f61fde6d7edfaf40c6bf103a5c4769c1" alt="Sitecore JSS Components"
In Sitecore SXA, there is an out-of-the-box Snippet component to cover such cases.
data:image/s3,"s3://crabby-images/67221/67221aa53c50488b6e025af212224dc61ef93b34" alt="SXA snippet"
Since our project was a combination of Sitecore SXA and JSS, we couldn’t use the SXA’s Snippet component. We had to develop a custom one.
Solution
The following idea came up to my mind.
- Create a simple JSS Component (aka Snippet) with a snippet-content placeholder inside.
- Create a new JSS Layout specifically to build a Snippet component structure separately.
- Put the Snippet Component (built in step 1) on the main JSS Layout (used for content pages). Select the Snippet Item created separately using the JSS Layout in step 2 as a data source.
- At runtime during the rendering process, check if the rendered item contains a Snippet Component somewhere. If YES — get a Snippet DataSouce, render it and put the rendered result in the snippet-content placeholder.
Implementation
- The first step is quite simple. In our project, we used the Sitecore First approach mode to build JSS components.
In Sitecore, create a Snippet JSON Rendering:
data:image/s3,"s3://crabby-images/915b0/915b041d3ae1f4fe18ece0076c3e86771d666d1b" alt="Snippet Rendering"
As for the DataSource, create a Snippet Item template inherited from:
* Base App Route template (Path: /sitecore/templates/Foundation/JSS Experience Accelerator/Multisite/Base App Route)
data:image/s3,"s3://crabby-images/49b60/49b6097583bbf535adb20e32131601ae50817831" alt="Base App Route Template"
* Global Datasource Behavior template (Path: /sitecore/templates/Foundation/Experience Accelerator/Local Datasources/Global Datasource Behavior)
data:image/s3,"s3://crabby-images/b2052/b20526a66bf782842a921c249fd299ad2451431d" alt="Global DS Template"
In the created Snippet rendering, put a data source link referenced to the Snippet Item template.
data:image/s3,"s3://crabby-images/8a6e3/8a6e33bd45a6ef49e28ae81df4cb583d33ca3cab" alt="Snippet DS Locations-1"
Then, create a Snippet JSS Component with a placeholder:
import React from 'react';
import { isExperienceEditorActive, Placeholder, Image } from '@sitecore-jss/sitecore-jss-react';
const SnippetComponent = (props) => {
if (!props.fields) {
return <h1 className="alarm">Datasource isn't set.<h1>;
}
return (
<h1 className="alarm">
Data is not provided. Contact administrators, please.
</h1>
);
}
return(
<>
<Placeholder name="snippet-content" rendering={props.rendering} />
</>
);
}
export default SnippetComponent;
2. Create an additional Layout that will allow you to configure the Snippet data source separately.
In Sitecore, create a new JSS Snippet Layout (just a copy of the existing one) with only one placeholder (in my case snippet-content):
data:image/s3,"s3://crabby-images/0d40b/0d40b88854693bd1e828d24fb41c0b69886c367d" alt="Snippet Layout"
In JSS, add the following piece of code to the Layout.js file:
import React from 'react';
import { Placeholder, VisitorIdentification,withSitecoreContext } from '@sitecore-jss/sitecore-jss-react';
import Helmet from 'react-helmet';
import './assets/app.css';
const Layout = ({ route, context }) => {
return (<React.Fragment>
<Helmet>
<title>
{(route.fields route.fields["Page Title"] route.fields["Page Title"].value) || route.displayName }
</title>
</Helmet>
<VisitorIdentification />
<Placeholder name="jss-top" rendering={route} />
<section className="site-wrapper">
<Placeholder name="jss-main" rendering={route} />
</section>
<Placeholder name="jss-footer" rendering={route} />
{context.pageEditing <Placeholder name="snippet-content" rendering={route} />}
</React.Fragment>
)};
export default Layout;
In step 1, we created the Snippet Item inherited from the Base App Route JSS Experience Accelerator template. It means we can use this item as a content page. Make sure the Snippet Item is referenced now to the Snippet Layout we created above.
data:image/s3,"s3://crabby-images/d58c2/d58c2f708b3d3d0eee3ce2ef0b3843e00b7f6d9d" alt="Snippet Layout Reference"
3. Find the demonstration below in the How To Use section.
4. After some investigations, I found out that the main entry point where it is possible to inject our custom code is the Render virtual method in LayoutService class (implementation of the ILayoutService Interface, Sitecore.LayoutService.dll).
data:image/s3,"s3://crabby-images/4d48a/4d48a563f7ba7e88d4f250da5638aa402dad5709" alt="Sitecore Layout Service"
So, to achieve our goal, we need to develop a custom Layout Service inherited from default Sitecore LayoutService and override the Render method to add a logic to find and render Snippets with its data source added on a page:
public class CustomLayoutService : LayoutService
{
public CustomLayoutService(
IPlaceholderRenderingService placeholderService,
ILayoutServiceContext serviceContext) : base(placeholderService, serviceContext)
{
}
public override RenderedItem Render(
Item item,
IRenderingConfiguration renderingConfiguration,
RenderOptions renderOptions = null)
{
return CustomRenderedItem(item, renderingConfiguration, renderOptions: renderOptions);
}
You can find the CustomRenderedItem method implementation here.
To inject the CustomLayoutService instead of the default one, we need to register the following dependencies:
public class RegisterDependencies : IServicesConfigurator
{
public void Configure(IServiceCollection serviceCollection)
{
serviceCollection.AddSingleton <IConfiguration, Configuration>();
serviceCollection.AddSingleton <IFieldRenderer, FieldRenderer>();
serviceCollection.AddSingleton <ISerializerService, SerializerService>();
serviceCollection.AddSingleton <IRenderJsonRenderingPipeline, RenderJsonRenderingPipeline>();
serviceCollection.AddSingleton <IGetFieldSerializerPipeline, GetFieldSerializerPipeline>();
serviceCollection.AddSingleton <IPlaceholderRenderingService, PlaceholderRenderingService>();
serviceCollection.AddSingleton <ILayoutServiceContext, PipelineLayoutServiceContext>();
serviceCollection.AddSingleton <IGetLayoutServiceContextPipeline, GetLayoutServiceContextPipeline>();
serviceCollection.AddSingleton <ILayoutService, CustomLayoutService>();
}
}
and apply the following patch:
<configuration xmlns:patch="http://www.sitecore.net/xmlconfig/" xmlns:role="http://www.sitecore.net/xmlconfig/role/">
<sitecore>
<services>
<configurator type="Sitecore.LayoutService.RegisterDependencies, Sitecore.LayoutService">
<patch:attribute name="type">JssSnippet.Configurators.RegisterDependencies, JssSnippet</patch:attribute>
</configurator>
</services>
</sitecore>
</configuration>
How To Use
- Create a Snippet item under Data folder of your SXA JSS Site (/sitecore/content/[JSS Tenant]/[JSS Site]/Data/Snippets/Demo/Snippet):
data:image/s3,"s3://crabby-images/2664a/2664a3ff3ab2672643c3ba4352cdb9259e088f46" alt="Snippet Item"
2. Open created Snippet Item in Experience Editor to add microcomponents there:
data:image/s3,"s3://crabby-images/ee08d/ee08d11d437cc34e301b2ec32790de6bbde4f6ec" alt="Empty Snippet Exp Editor"
3. Add JSS Components you would like to use as a snippet then. For instance, in my case, I am going to add the Section component, then Grid component, then Column component (2), then Image component (left column), then RTE component (right column):
Section:
data:image/s3,"s3://crabby-images/db649/db6495d7572c2a5b460b2048a12218d2784d58c2" alt="Section"
Grid with two Columns:
data:image/s3,"s3://crabby-images/50600/50600020da7c184a1cd898be604fcedce249e24d" alt="Grid"
The image in the Left Column:
data:image/s3,"s3://crabby-images/6c843/6c843f28b124bd780e5ed9cc8522ea0cb7900b8a" alt="Snippet-2"
Rich Text in the Right Column:
data:image/s3,"s3://crabby-images/ce3c7/ce3c75f7d3af9db7039526b048e7eb3ba82f634f" alt="Sitecore Containers"
Ok, now we have our Snippet data source with components inside:
data:image/s3,"s3://crabby-images/4ac1e/4ac1e5ae36869a209956921c5d0175c1ce0c4467" alt="Snippet Item Full"
4. Now, it is time to add the Snippet to a site page. Open a content page in Experience Editor and add the Snippet Component in any placeholder you need:
data:image/s3,"s3://crabby-images/acc04/acc045507fdec50717ceb5f431f7e06f19b73312" alt="Add Snippet"
And here is the result — we’ve got a Snippet component with Microcomponents inside.
data:image/s3,"s3://crabby-images/3ad44/3ad441015b6a226ae4ae03a788d4d6f1d1e861da" alt="Snippet On Page"
5. Let’s put the same snippet on the Top placeholder on the same page.
data:image/s3,"s3://crabby-images/66cc4/66cc4c1bd81d4468029b98c187cead6c167d96ba" alt="Snippet on top placeholder"
And now we have a reusable feature — two Snippets on the page.
data:image/s3,"s3://crabby-images/22009/22009348f581b419352c822d94af3817da39415d" alt="Two Snippets"
Let’s check the presentation details for this page.
data:image/s3,"s3://crabby-images/f9792/f979296a93284ac2af4144c115707f581a4e50b4" alt="Presentation Details"
6. There is also one feature in the SXA’s Snippet, such as Global Data Source Behavior. Our custom Snippet item also has this approach (inherited from the SXA template — /sitecore/templates/Foundation/Experience Accelerator/Local Datasources/Global Datasource Behavior)
data:image/s3,"s3://crabby-images/cde61/cde6130c365e6a7c8c825ca9fd2944889a543716" alt="DataSource Behavior"
7. Probably, you have the following questions:
* Is it possible to Edit components that are inside the snippet directly on a content page? — Yes, it is.
* Is it possible to Put another snippet to an already created one? — Yes, it is. As of now, there is only one restriction — you are not able to put the same Snippet inside.
* Do the Sitecore Personalization Rules work with Snippets? — Yes, they do.
* Can I contribute to it? — Sure, just make a fork and help me improve it.
That’s all for today. You can find the implementation in my GitHub.
Feel free to contribute.
Happy coding, Sitecorians!
Very nice 🙂
Goran, thank you for the feedback!