Hide builder content from visitors who have not consented to tracking

The activity tracking documentation explains that Xperience logs the Form submit activity automatically, even for users who have not consented to tracking. As the page notes, this means you need to ensure that usage of personal data collected through such activities falls under legitimate interest purposes.

To avoid potential legal complications, you may want to display certain forms only to users who have consented to tracking.

This guide covers the process of creating a Page Builder section which hides its contents from users who have not given consent for tracking. The code you develop will add an alternate version of the existing Form column section included in the training guides repository.

The code in this guide relies on the value of the CMSCookieLevel cookie, and assumes this has been properly mapped to consent in your application. 

You can find an example of this in the data protection series of training guides.

Activity tracking series

This guide is part of a series on Activity tracking.

If you want to follow along, you can start here.

Activity tracking functionality ties in with consents, which are covered in detail in the Data protection series.

Before you start

This guide requires the following:

The examples in this guide require that you:

Code samples

You can find a project with completed, working versions of code smaples from this guide and others in the finished branch of the Training guides repository.

The main branch of the repository provides a starting point to code along with the guides.

The code samples in this guide are for .NET 8 only.

They come from a project that uses implicit using directives. You may need to add additional using directives to your code if your project does not use this feature.

Edit the view model

This form section needs to execute complex logic and change its behavior accordingly, so you should base it on a view component.

A view component will allow you to pass your own model to the section, rather than relying on the ComponentViewModel<TPropertyModel> object Xperience provides for basic sections.

You need to create this view model before you can use it in the view component.

  1. Navigate to TrainingGuides.Web/Features/Shared/Sections/FormColumn/FormColumnSectionViewModel.cs.
  2. Add a boolean field to represent whether the form section should render its content.

The SectionAnchor property corresponds to a property of the same name from the existing FormColumnSectionProperties class. It lets editors specify the value of an id attribute added to the HTML <section> tag that wraps the contents of the FormColumnSection.

C#FormColumnSectionViewModel.cs


namespace TrainingGuides.Web.Features.Shared.Sections.FormColumn;

public class FormColumnSectionViewModel
{
    public bool ShowContents { get; set; }
    public string SectionAnchor { get; set; } = string.Empty;
}

Alter the view

You must account for the new ShowContents property of the FormColumnSectionViewModel in the view, used to conditionally display its contents.

  1. Navigate to TrainingGuides.Web/Features/Shared/Sections/FormColumn/FormColumnSection.cshtml.

  2. Add a conditional that only shows the view’s contents if the ShowContents property of the model is true, or if the page is in Preview or Page Builder mode.

    Checking for preview mode ensures that the section contents will be visible in Page Builder regardless of an editor’s cookie level.

cshtmlFormColumnSection.cshtml


@using TrainingGuides.Web.Features.Shared.Sections.FormColumn
@model FormColumnSectionViewModel

@if(Model.ShowContents || Context.Kentico().Preview().Enabled)
{
    var sectionAnchor = !string.IsNullOrWhiteSpace(Model.SectionAnchor) ? $"id={Model.SectionAnchor}" : "";

    <section class="c-section form" @sectionAnchor>
        <div class="container">
            <div class="row justify-content-center">
                <div class="col-lg-8">
                    <widget-zone />
                </div>
            </div>
        </div>
    </section>
}

Create a new view component and tweak the existing one

The first guide in this series explained how you can use the value of the CMSCookieLevel cookie to determine whether the current visitor is trackable, and showed you how to create a method called CurrentContactCanBeTracked to determine whether or not Xperience should track the current user.

Use this method in a new view component to create a view model and pass it along to the view.

  1. Add a new view component called FormColumnSectionConsentViewComponent.cs to the TrainingGuides.Web/Features/Shared/Sections/FormColumn folder.

  2. Inject an ICookieConsentService instance.

  3. Set the Invoke method to take a ComponentViewModel<FormColumnSectionProperties> object as a parameter.

    Xperience will supply this object automatically. You can use it to access properties configured by editors from the Invoke method.

  4. Use sectionProperties.Properties.SectionAnchor and CurrentContactCanBeTracked to populate the properties of a new FormColumnSectionConsentViewModel, and pass it along to the view.

  5. Create a new constant to hold the identifier and use it to register the view component as a Page Builder section.

    C#FormColumnSectionConsentViewComponent.cs
    
     using Kentico.PageBuilder.Web.Mvc;
     using Microsoft.AspNetCore.Mvc;
     using TrainingGuides.Web.Features.Shared.Sections.FormColumn;
     using TrainingGuides.Web.Features.DataProtection.Services;
    
     [assembly: RegisterSection(
         identifier: FormColumnSectionConsentViewComponent.IDENTIFIER,
         viewComponentType: typeof(FormColumnSectionConsentViewComponent),
         name: "Form column: Consent-based",
         propertiesType: typeof(FormColumnSectionProperties),
         Description = "Form column section that hides its contents if the visitor has not consented to tracking.",
         IconClass = "icon-square")]
    
     namespace TrainingGuides.Web.Features.Shared.Sections.FormColumn;
    
     public class FormColumnSectionConsentViewComponent : ViewComponent
     {
         public const string IDENTIFIER = "TrainingGuides.FormColumnSectionConsent";
    
         private readonly ICookieConsentService cookieConsentService;
    
         public FormColumnSectionConsentViewComponent(ICookieConsentService cookieConsentService)
         {
             this.cookieConsentService = cookieConsentService;
         }
    
         public IViewComponentResult Invoke(ComponentViewModel<FormColumnSectionProperties> sectionProperties)
         {
             //If the CMSCookieLevel is set to All (1000) or higher, and Data Protection is set up, it means the visitor has the appropriate consent level for tracking.
             bool showContents = cookieConsentService.CurrentContactCanBeTracked();
    
             var model = new FormColumnSectionViewModel()
             {
                 SectionAnchor = sectionProperties.Properties.SectionAnchor,
                 ShowContents = showContents
             };
    
             return View("~/Features/Shared/Sections/FormColumn/FormColumnSection.cshtml", model);
         }
     }
     
  6. Adjust the existing FormColumnSectionViewComponent.cs file to account for the new ShowContents property of the view model.

    C#FormColumnSectionViewComponent.cs
    
     ...
     public IViewComponentResult Invoke(ComponentViewModel<FormColumnSectionProperties> sectionProperties)
     {
         var model = new FormColumnSectionViewModel()
         {
             SectionAnchor = sectionProperties.Properties.SectionAnchor,
             ShowContents = true
         };
         return View("~/Features/Shared/Sections/FormColumn/FormColumnSection.cshtml", model);
     }
     ...
     

Add an identifier

Now, in case the identifier of the new consent-based component needs to be accessed externally, add it to the component identifiers file.

Add a new constant to the Sections class in TrainingGuides.Web/ComponentIdentifiers.cs that references the constant from the view component.

C#ComponentIdentifiers.cs


...
public static class ComponentIdentifiers
{
    public static class Sections
    {
        ...
        public const string FORM_COLUMN_CONSENT = FormColumnSectionViewComponent.IDENTIFIER;
        ...
    }
    ...
}

What’s next

You have implemented a layout and logic that displays or hides content on the website depending on a visitor tracking consent. In order to see it in action, you need to use it on a page.

  1. Run your project locally, login to the administration interface, and navigate to the Training guides pages channel.

  2. Click Edit → Create new version to edit the Contact-us page.

  3. Set Page Builder to use the Form column: Consent-based section, then Save and Publish the web page.

  4. Now pretend that you are a visitor and navigate to your website in a browser’s incognito window. Visit the Cookie policy page, set the cookie level to essential and then visit your Contact us page. The section should not display.

  5. Go back to the Cookie policy page, change your consent to Marketing and revisit Contact us page. If you’ve implemented the functionality properly, you’ll see the form displayed on the page.

Your site now includes a section with a powerful custom logic. Your website editors can use the new section to hide forms on the website from specific visitors. But they are not limited to just forms. They can hide any custom widgets which might lead to, for example, a GDPR compliance breach if displayed for all the visitors.