Kentico Xperience 13.0 custom module to protect pages referenced in code from deletion and shared abstractions for linkable pages.
This package is compatible with .NET Standard 2.0 libraries integrated with Kentico Xperience 13.0.
Kentico Xperience 13.0 ASP.NET Core Tag Helper that generates links to predefined pages using their NodeGUID values, and extension methods for registering dependencies in ASP.NET Core.
This package is compatible with ASP.NET Core 3.1+ applications or libraries integrated with Kentico Xperience 13.0.
-
Install the
XperienceCommunity.LinkablePages
NuGet package in a class library shared by your ASP.NET Core and CMS applications:dotnet add package XperienceCommunity.LinkablePages
-
In the shared class library, create a class implementing the
ILinkablePage
where you define pages that are available to the tag helper:Populate the
Guid
values with theNodeGUID
of each page in the content tree that you need to link to in your application.using XperienceCommunity.LinkablePages;
public namespace Sandbox.Shared { public class LinkablePage : ILinkablePage { public static LinkablePage Home { get; } = new LinkablePage(new Guid("...")); public static LinkablePage Store { get; } = new LinkablePage(new Guid("...")); public static LinkablePage ContactUs { get; } = new LinkablePage(new Guid("...")); public static LinkablePage TermsOfUse { get; } = new LinkablePage(new Guid("...")); public Guid NodeGUID { get; } protected LinkablePage(Guid nodeGUID) => NodeGUID = nodeGUID; public static IReadOnlyList<LinkablePage> All { get; } = new List<LinkablePage> { Home, Store, ContactUs, TermsOfUse }; } }
-
In the shared class library, create an implementation of the
ILinkablePageInventory
interface, which will be used to determine which Pages in the application should be protected:using XperienceCommunity.LinkablePages;
public class LinkablePageInventory : ILinkablePageInventory { public bool IsLinkablePage(TreeNode page) { return LinkablePage.All.Any(p => p.NodeGUID == page.NodeGUID); } }
-
Install the
XperienceCommunity.PageLinkTagHelpers
NuGet package in your ASP.NET Core project:dotnet add package XperienceCommunity.PageLinkTagHelpers
-
Add the
@addTagHelper
directive to your_ViewImports.cshtml
file:(optional) Add your
LinkablePage
class's namespace to your_ViewImports.cshtml
file (e.g.,Sandbox.Shared
).// Add this using to make LinkablePages easy to access in Views @using Sandbox.Shared @addTagHelper *, Microsoft.AspNetCore.Mvc.TagHelpers @addTagHelper *, Kentico.Content.Web.Mvc @addTagHelper *, Kentico.Web.Mvc @addTagHelper *, DancingGoatCore // Add this directive to use the Tag Helper @addTagHelper *, XperienceCommunity.PageLinkTagHelpers
-
Register the library with ASP.NET Core DI:
using XperienceCommunity.LinkablePages;
public void ConfigureServices(IServiceCollection services) { // Use standard registration services .AddXperienceCommunityPageLinks() .AddXperienceCommunityPageLinksProtection<LinkablePageInventory>(); // or use a custom implementation of ILinkablePageLinkRetriever services .AddXperienceCommunityPageLinks<MyCustomLinkablePageLinkRetriever>() .AddXperienceCommunityPageLinksProtection<LinkablePageInventory>(); }
-
Add the data protection custom module registration to your ASP.NET Core application (in
Startup.cs
or wherever you register your dependencies):using XperienceCommunity.LinkablePages; [assembly: RegisterModule(typeof(LinkablePageProtectionModule))]
-
Use the
xp-page-link
tag helper in an<a>
element in a Razor View:<a href="" xp-page-link="LinkablePage.Home"> <img src="/getmedia/10d5e094-d9aa-4edf-940d-098ca69b5f77/logo.png" alt="..." /> </a>
-
Create a custom module class in your CMS application to register the
LinkablePageProtectionModule
and theILinkablePageInventory
implementation:using XperienceCommunity.LinkablePages;
// Registers this custom module class [assembly: RegisterModule(typeof(DependencyRegistrationModule))] // Registers the library's custom module class [assembly: RegisterModule(typeof(LinkablePageProtectionModule))] namespace CMSApp { public class DependencyRegistrationModule : Module { public DependencyRegistrationModule() : base(nameof(DependencyRegistrationModule)) { } protected override void OnPreInit() { base.OnPreInit(); // Registers the ILinkablePageInventory implementation used by the LinkablePageProtectionModule Service.Use<ILinkablePageInventory, LinkablePageInventory>(); } } }
-
Use Kentico Xperience's Content Staging to keep your
LinkablePage
NodeGUID
values valid between different environments.
Razor (assuming the ContactUs Page has a DocumentName
of "Contact Us"):
<a href="" xp-page-link="LinkablePage.ContactUs"></a>
Generated HTML will set the child content, href
and title
attributes:
<a href="/contact-us" title="Contact Us">Contact Us</a>
Razor (assuming the ContactUs Page has a DocumentName
of "Contact Us"):
<a href="" title="Please contact us" xp-page-link="LinkablePage.ContactUs">
<img src="..." />
</a>
Generated HTML will keep the child content and only set the href
and title
attributes:
<a href="/contact-us" title="Please contact us">
<img src="..." />
</a>
Razor:
<a href="" title="" xp-page-link="LinkablePage.ContactUs">
No title necessary!
</a>
Generated HTML will not populate the title
attribute, assuming the child content provides some text:
<a href="/contact-us" title=""> No title necessary! </a>
Razor (assuming the ContactUs Page has a DocumentName
of "Contact Us"):
<a href="" title="" xp-page-link="LinkablePage.ContactUs"> </a>
Generated HTML will populate the title
for accessibility:
<a href="/contact-us" title="Contact Us"> Contact Us </a>
Razor:
<a
href=""
xp-page-link="LinkablePage.ContactUs"
title="Please contact us"
xp-page-link-query-params="@(new() { { "parameter": "value" } })">
Contact us for help!
</a>
Generated HTML will include query string parameters in the href
, and set the title
attribute/child content as appropriate:
<a href="/contact-us?parameter=value" title="Please contact us">
Contact us for help!
</a>
If you discover a problem, please open an issue.
If you would like contribute to the code or documentation, please open a pull request.