Register content location with your custom widgets

By default, all built-in and dynamic Sitefinity CMS widgets, which can display content items, register the items’ locations and generate the items’ links using the ContentLocationService. When you place such widget on a page and publish the page, each widget sends data to Sitefinity CMS. The data contains the items that the widget displays, their type, and the page they appear on.

When you are using your own custom widgets to display Sitefinity CMS content items, you must manually register for the ContentLocationService. This way, Sitefinity CMS will know the locations of the content items displayed with you custom widget. 

When you publish a page, Sitefinity CMS check for all widgets on the page that implement the IContentLocatableView interface, and for each of them it gets the location information.

If you want your custom widget to have a canonical URL and to provide a location for items, you must implement the IContentLocatableView interface.

NOTE: Sitefinity CMS supports using the IContentLocatableView interface only with custom widgets that inherit from  SimpleViewor SimpleScriptView 

Implement the IContentLocatableView interface

The IContentLocatableView interface defines the common information that must be provided by each widget that is able to show content items. It is used for the preview of content items and for issuing a canonical link element for SEO. 

You implement the interface, by implementing the IEnumerable<IContentLocationInfo> GetLocations() method  and the DisableCanonicalUrlMetaTag property.

The GetLocations() method is called by Sitefinity CMS upon page publish and returns the specific location information about the item type and provider that is used to display the items. In your implementation of the GetLocations() method you must construct an object of type ContentLocationInfo, which tells Sitefinity CMS what type of content is displayed on the page, and what is the full name of the provider used for this content type.

The DisableCanonicalUrlMetaTag property requires no additional implementation by default. It is responsible for instructing Sitefinity CMS whether to render a canonical link. You can add some additional logic to determine its value inside the property implementation or set the value correspondingly.

The following code sample demonstrates the implementation of the IContentLocatableView in a custom widget that inherits from SimpleView:

using System;
using System.Linq;
using System.Web.UI.WebControls;
using Telerik.Sitefinity.ContentLocations;
using Telerik.Sitefinity.DynamicModules;
using Telerik.Sitefinity.Utilities.TypeConverters;
using Telerik.Sitefinity.Web.UI;
using Telerik.Sitefinity.Model;
namespace SitefinityWebApp.IContentLocatableWidget
{
public class SampleWidget_IContentLocatableView_Impl : SimpleView, IContentLocatableView
{
#region Properties
/// <summary>
/// Gets or sets the message that will be displayed in the label.
/// </summary>
public string Message { get; set; }
/// <summary>
/// Obsolete. Use LayoutTemplatePath instead.
/// </summary>
protected override string LayoutTemplateName
{
get
{
return string.Empty;
}
}
/// <summary>
/// Gets the layout template's relative or virtual path.
/// </summary>
public override string LayoutTemplatePath
{
get
{
if (string.IsNullOrEmpty(base.LayoutTemplatePath))
return layoutTemplatePath;
return base.LayoutTemplatePath;
}
set
{
base.LayoutTemplatePath = value;
}
}
#endregion
#region Control References
/// <summary>
/// Reference to the Label control that shows the Message.
/// </summary>
protected virtual Label MessageLabel
{
get
{
return this.Container.GetControl<Label>("MessageLabel", true);
}
}
/// <summary>
/// Reference to the Label control that shows the Message.
/// </summary>
protected virtual Repeater Repeater1
{
get
{
return this.Container.GetControl<Repeater>("RepeaterPressReleases", true);
}
}
#endregion
#region Methods
/// <summary>
/// Initializes the controls.
/// </summary>
/// <param name="container"></param>
/// <remarks>
/// Initialize your controls in this method. Do not override CreateChildControls method.
/// </remarks>
protected override void InitializeControls(GenericContainer container)
{
DynamicModuleManager dynamicModuleManager = DynamicModuleManager.GetManager();
Type pressReleaseType = TypeResolutionService.ResolveType(this.DynamicContentTypeName);
// This is how we get the collection of Press release items
var myCollection = dynamicModuleManager.GetDataItems(pressReleaseType)
.Where(c => c.Visible && c.Status == Telerik.Sitefinity.GenericContent.Model.ContentLifecycleStatus.Live)
.OrderBy(x => x.GetValue<String>("Title")).Skip(0).Take(this.PageSize);
this.Repeater1.DataSource = myCollection;
this.Repeater1.DataBind();
Label messageLabel = this.MessageLabel;
if (string.IsNullOrEmpty(this.Message))
{
messageLabel.Text = "Hello, World!";
}
else
{
messageLabel.Text = this.Message;
}
}
#endregion
#region Private members & constants
public static readonly string layoutTemplatePath = "~/IContentLocatableWidget/SampleWidget_IContentLocatableView_Impl.ascx";
#endregion
//add IContentLocatableInfo implementation requirements here
public bool? DisableCanonicalUrlMetaTag { get; set; }
public System.Collections.Generic.IEnumerable<IContentLocationInfo> GetLocations()
{
var location = new ContentLocationInfo();
var contentType = Telerik.Sitefinity.Utilities.TypeConverters.TypeResolutionService.ResolveType(this.DynamicContentTypeName);
location.ContentType = contentType;
string providerName = string.Empty;
var manager = DynamicModuleManager.GetManager(providerName);
var fullProviderName = manager.Provider.Name;
location.ProviderName = fullProviderName;
yield return location;
}
public string DynamicContentTypeName
{
get
{
return "Telerik.Sitefinity.DynamicTypes.Model.Samples.Sample";
}
}
private int pageSize;
public int PageSize
{
get
{
if (this.pageSize == 0)
{
this.pageSize = 5;
}
return pageSize;
}
set { pageSize = value; }
}
}
}

And its corresponding widget template:

Register a ContentLocation for widgets that display multiple content types

In case your custom widget displays items from more than one content type, you must specify all content types when you register a ContentLocation. To achieve that, you need to return a list of  ContentLocationInfo objects, containing a separate ContentLocationInfo object for each content type displayed by your widget.  

The following sample demonstrates returning a list of  ContentLocationInfo objects in the GetLocations method implementation:

Use a ContentLocationFilter to adjust the registered ContentLocation information

By default, when you register a content location with your custom widget, you inform Sitefinity CMS that all items from the specified content type and respective provider are displayed by this widget. However, there are certain scenarios where you need to limit the scope of displayed items (for example show only items that meet certain criteria like a taxonomy, publication date range, and so on). In such cases you must adjust the information you are passing with your ContentLocation, so you inform Sitefinity CMS that only certain items from the specified type can be found on the registered location. This is done via configuring a ContentLocationFilter. A ContentLocationFilter is an implementation of the IContentLocationFilter interface and represents a string filter applied to the content location. You must specify a Value and an optional Name properties for the filter. The Value represents a FilterExpression which is parsed by Sitefinity CMS when working with this location. For more information about constructing a FilterExpression for Sitefinity CMS content see: Filter expressions for content items

The following sample demonstrates adding a ContentLocationFilter to the sample referenced earlier in this article. The filter informs the Sitefinity CMS ContentLocationService that this custom widget displays only content items which are marked with a taxon.

using System;
using Telerik.Sitefinity.ContentLocations;
using Telerik.Sitefinity.DynamicModules;
using Telerik.Sitefinity.Web.UI;
namespace SitefinityWebApp
{
public class RegisterContentLocationWithFilter : SimpleView, IContentLocatableView
{
protected override void InitializeControls(GenericContainer container)
{
throw new NotImplementedException();
}
public bool? DisableCanonicalUrlMetaTag { get => throw new NotImplementedException(); set => throw new NotImplementedException(); }
public string DynamicContentTypeName
{
get
{
return "Telerik.Sitefinity.DynamicTypes.Model.Samples.Sample";
}
}
public System.Collections.Generic.IEnumerable<IContentLocationInfo> GetLocations()
{
var contentType = Telerik.Sitefinity.Utilities.TypeConverters.TypeResolutionService.ResolveType(DynamicContentTypeName);
string providerName = string.Empty;
var manager = DynamicModuleManager.GetManager(providerName);
var fullProviderName = manager.Provider.Name;
//Register a content location
var location = new ContentLocationInfo();
//Inform Sitefinity CMS that this widget displays content of this type
location.ContentType = contentType;
// And this provider
location.ProviderName = fullProviderName;
//Build a filter expression
var filterExpression = "Tags.Contains((a82b9e17-13a1-405d-b4a3-4f301437bec6))";
if (!string.IsNullOrEmpty(filterExpression))
{
//Add a ContentLocationFilter to the ContentLocation
location.Filters.Add(new BasicContentLocationFilter(filterExpression));
}
yield return location;
}
}
}

 

Increase your Sitefinity skills by signing up for our free trainings. Get Sitefinity-certified at Progress Education Community to boost your credentials.

Get started with Integration Hub | Sitefinity Cloud | Sitefinity SaaS

This free lesson teaches administrators, marketers, and other business professionals how to use the Integration hub service to create automated workflows between Sitefinity and other business systems.

Web Security for Sitefinity Administrators

This free lesson teaches administrators the basics about protecting yor Sitefinity instance and its sites from external threats. Configure HTTPS, SSL, allow lists for trusted sites, and cookie security, among others.

Foundations of Sitefinity ASP.NET Core Development

The free on-demand video course teaches developers how to use Sitefinity .NET Core and leverage its decoupled architecture and new way of coding against the platform.

Was this article helpful?