World is now on Opti ID! Learn more

Jens Qvist
Dec 19, 2011
  2664
(0 votes)

Creating a BrowserLanguageCriteria for Visitor groups.

This is my first blog post ever, on any site I think! So I’m a bit nervous, but I hope that someone will find it interesting. So here goes!

This morning I decided that I should learn how to create custom criterias for Visitor groups. I started out by reading an article by Ted Nyberg, to get an idea on how it worked.

But I was short on ideas on what to build. Until I stumbled on http://criteriapack.codeplex.com/, where they had a list on planned criterias. Perfect I thought, and choose to create the BrowserLanguageCriterion. I used BrowserOSCriterion as a template for my own code, and started working.

language

So first I needed a settings class as mentioned in Teds article. I decided that the editor would want to choose a language from a dropdownlist and every culture/language in the browser should be availible. It was easy figuring out how to get all cultures. The problem for me was to get all cultures into an enum, since I started out using EnumSelectionFactory. Which I guess require an enum?

[DojoWidget(SelectionFactoryType = typeof(EnumSelectionFactory), 
        AdditionalOptions = "{ selectOnClick: true }")]
public string BrowserLanguage { get; set; }

To be honest, I haven’t used Enums much and had to google if there was a possibility to convert a list/array of languages to an enum. Apparently not, according to some forum post.

My solution was kinda easy once I figured out how to do it. I created my own SelectionFactoryType.

public class LanguageFactory : ISelectionFactory
{
    public IEnumerable<SelectListItem> GetSelectListItems(Type property)
    {
        //For each culture availible
        foreach(CultureInfo ci in CultureInfo.GetCultures(CultureTypes.AllCultures)) {
            yield return new SelectListItem() { Text = ci.EnglishName, Value = ci.EnglishName };
        }  
    }
}
and changed my SelectionFactoryType to my newly created one;
[Required]
[DojoWidget(SelectionFactoryType = typeof(LanguageFactory))]
public string BrowserLanguage { get; set; }
Finished with my settings class, I started working on the criterion class. Basically what I needed was to compare the
browserlanguage with the language choosen in the criteria for the visitorgroup.
This was done easily by matching the two.
[VisitorGroupCriterion(
    Category = "Technical Criteria",
    Description = "Match Browser language with specified value",
    DisplayName = "Browser language")]
public class BrowserLanguageCriterion : CriterionBase<BrowserLanguageModel>
{
    private string _browserlang;
    //matches the browserlanguage with the criterion set language.
    public override bool IsMatch(System.Security.Principal.IPrincipal principal, 
System.Web.HttpContextBase httpContext)
    {
        return ((String.IsNullOrEmpty(base.Model.BrowserLanguage) ? true : 
        StringMatchHelper.IsMatch(_browserlang, base.Model.BrowserLanguage, 
        base.Model.BrowserLanguageMatchType)));             
    }

    public override void Subscribe(ICriterionEvents criterionEvents)
    {
        base.Subscribe(criterionEvents);
        criterionEvents.StartRequest += criterionEvents_StartRequest;
    }

    private void criterionEvents_StartRequest(object sender, CriterionEventArgs e)
    {
        //Get the browserlanguage
        string browserLanguage = e.HttpContext.Request.UserLanguages[0];
        Thread.CurrentThread.CurrentCulture = 
        System.Globalization.CultureInfo.CreateSpecificCulture(browserLanguage);
        _browserlang = Thread.CurrentThread.CurrentCulture.EnglishName;
    }

    public override void Unsubscribe(ICriterionEvents criterionEvents)
    {
        criterionEvents.StartRequest -= criterionEvents_StartRequest;
        base.Unsubscribe(criterionEvents);
    }
}

It was a great experience getting this to work and it made me write my first blog!
I’m quite pleased how this day turned out and hope that someone will find my
blogpost interesting!
/Jens Qvist Here is the full source;

using EPiServer.Personalization.VisitorGroups;
using EPiServer.Data;
using System.Threading;
using System.Globalization;
using EPiServer.Web.Mvc.VisitorGroups;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.Web.Mvc;
using System;

namespace EPiServer.Classes
{
    public class BrowserLanguageModel : CriterionModelBase
    {
        private MatchStringType _browserLanguageMatchType;
    
        [DojoWidget(SelectionFactoryType = typeof(EnumSelectionFactory), 
             AdditionalOptions = "{ selectOnClick: true }")]
        public MatchStringType BrowserLanguageMatchType
        {
            get
            {
                return this._browserLanguageMatchType;
            }
            set
            {
                this._browserLanguageMatchType = value;
            }
        }
        [Required]
        [DojoWidget(SelectionFactoryType = typeof(LanguageFactory))]
        public string BrowserLanguage { get; set; }

        public override ICriterionModel Copy()
        {
            return base.ShallowCopy();
        }
    }

    [VisitorGroupCriterion(
        Category = "Technical Criteria",
        Description = "Match Browser language with specified value",
        DisplayName = "Browser language")]
    public class BrowserLanguageCriterion : CriterionBase<BrowserLanguageModel>
    {
        private string _browserlang;
        //matches the browserlanguage with the criterion set language.
        public override bool IsMatch(System.Security.Principal.IPrincipal principal, 
        System.Web.HttpContextBase httpContext)
        {
            return ((String.IsNullOrEmpty(base.Model.BrowserLanguage) ? true : 
            StringMatchHelper.IsMatch(_browserlang, base.Model.BrowserLanguage, 
            base.Model.BrowserLanguageMatchType)));             
        }
        public override void Subscribe(ICriterionEvents criterionEvents)
        {
            base.Subscribe(criterionEvents);
            criterionEvents.StartRequest += criterionEvents_StartRequest;
        }
        private void criterionEvents_StartRequest(object sender, CriterionEventArgs e)
        {
            //Get the browserlanguage
            string browserLanguage = e.HttpContext.Request.UserLanguages[0];
            Thread.CurrentThread.CurrentCulture = 
            System.Globalization.CultureInfo.CreateSpecificCulture(browserLanguage);
            _browserlang = Thread.CurrentThread.CurrentCulture.EnglishName;
        }
        public override void Unsubscribe(ICriterionEvents criterionEvents)
        {
            criterionEvents.StartRequest -= criterionEvents_StartRequest;
            base.Unsubscribe(criterionEvents);
        }
    }
    public class LanguageFactory : ISelectionFactory
    {
        public IEnumerable<SelectListItem> GetSelectListItems(Type property)
        {
            //For each culture availible
            foreach(CultureInfo ci in CultureInfo.GetCultures(CultureTypes.AllCultures)) {
                yield return new SelectListItem() { Text = ci.EnglishName, Value = ci.EnglishName };
            }  
        }
    }
}

Dec 19, 2011

Comments

Please login to comment.
Latest blogs
Make Global Assets Site- and Language-Aware at Indexing Time

I had a support case the other day with a question around search on global assets on a multisite. This is the result of that investigation. This co...

dada | Jun 26, 2025

The remote server returned an error: (400) Bad Request – when configuring Azure Storage for an older Optimizely CMS site

How to fix a strange issue that occurred when I moved editor-uploaded files for some old Optimizely CMS 11 solutions to Azure Storage.

Tomas Hensrud Gulla | Jun 26, 2025 |

Enable Opal AI for your Optimizely products

Learn how to enable Opal AI, and meet your infinite workforce.

Tomas Hensrud Gulla | Jun 25, 2025 |

Deploying to Optimizely Frontend Hosting: A Practical Guide

Optimizely Frontend Hosting is a cloud-based solution for deploying headless frontend applications - currently supporting only Next.js projects. It...

Szymon Uryga | Jun 25, 2025

World on Opti ID

We're excited to announce that world.optimizely.com is now integrated with Opti ID! What does this mean for you? New Users:  You can now log in wit...

Patrick Lam | Jun 22, 2025

Avoid Scandinavian Letters in File Names in Optimizely CMS

Discover how Scandinavian letters in file names can break media in Optimizely CMS—and learn a simple code fix to automatically sanitize uploads for...

Henning Sjørbotten | Jun 19, 2025 |