Skip to main content

Microsoft

Glass Mapper – Mapping Sitecore List Fields

Sitecore has five list field types that Glass Mapper maps: Checklist, Multilist, Multilist with Search, Treelist, and TreelistEx. Out of the box, Glass Mapper supports mapping these field types to two .NET collection types: IEnumerable<T> and IList<T>. In this post I’ll cover how to use Glass Mapper to map your code models to these field types.

Example Templates

For this post I’ll use two sample templates, Product and State, that look as follows:

  • Product
    • Name – Single-Line Text
    • EligibleStates
  • State
    • Name – Single-Line Text
    • Abbreviation – Single-Line Text

The Product template has an EligibleStates field that holds a list of States. I chose to make the EligibleStates field a Checklist, but the MultilistMultilist with SearchTreelist, or TreelistEx field types work too.

Interface Models

The code for the Product template looks like this:

[SitecoreType(TemplateId = "{00000000-0000-0000-0000-000000000001}", AutoMap = true)]
public interface IProduct
{
    Guid Id { get; set; }
    string Name { get; set; }
    IList<IState> EligibleStates { get; set; }
}

Note that the EligibleStates property could also be of type IEnumerable<IState> and Glass Mapper would map it. Other .NET collection types, such as Array<T>Collection<T>List<T>, etc., are not supported by Glass Mapper out of the box and will not be populated if present on your model.
The code for the State template looks like this:

[SitecoreType(TemplateId = "{00000000-0000-0000-0000-000000000002}", AutoMap = true)]
public interface IState
{
    Guid Id { get; set; }
    string Name { get; set; }
    string Abbreviation { get; set; }
}

Mapping from Sitecore in Controllers

To map list data from Sitecore in your controllers, just make the controller inherit from GlassController<T>, where T is the type of data source that your rendering uses (e.g., IProduct). You can then access the DataSource property to get your data from Sitecore. The EligibleStates property of the DataSource works as expected–all of the selected states in Sitecore are present.

public class ProductController : GlassController<IProduct>
{
    public ActionResult Product()
    {
        var product = DataSource;
        var viewModel = new ProductViewModel
        {
            Name = product.Name,
            EligibleStates = product.EligibleStates.Select(s => $"{s.Name} ({s.Abbreviation})")
        };
        return View(viewModel);
    }
}

Mapping from Sitecore, Elsewhere

Grabbing list data from Sitecore outside of your controllers is no problem. Just pull the item you want out of a SitecoreContext or SitecoreService and iterate over all the items in your List<T> or IEnumerable<T> property.

using (var sitecoreContext = new SitecoreContext())
{
    var product = sitecoreContext.GetItem<IProduct>("/sitecore/content/Products/Product1");
    var productName = product.Name;
    foreach (var state in product.EligibleStates)
    {
        var stateName = state.Name;
        var stateAbbr = state.Abbreviation;
    }
}

Modifying List Fields

To modify list fields, create a new instance of your item in Sitecore with the Create method on your SitecoreContext or SitecoreService, grab the items from Sitecore that you want to add to or remove from your list field, add or remove them, and save.

using (var sitecoreContext = new SitecoreContext())
{
    var georgia = sitecoreContext.GetItem<IState>("/sitecore/content/Settings/States/Georgia");
    var productsFolder = sitecoreContext.GetItem<Item>("/sitecore/content/Settings/Products");
    var product = sitecoreContext.Create(productsFolder, "New Product");
    product.Name = "New Product";
    product.EligibleStates.Add(georgia);
    sitecoreContext.Save(product);
}

Concrete Models

The code for the Product template looks like this:

[SitecoreType(TemplateId = "{00000000-0000-0000-0000-000000000001}", AutoMap = true)]
public class Product
{
    public virtual Guid Id { get; set; }
    public virtual string Name { get; set; }
    public virtual IList<IState> EligibleStates { get; set; }
}

Again, the EligibleStates property could also be of type IList<IState>. Other .NET collection types, such as Array<T>Collection<T>List<T>, etc., are not supported by Glass Mapper out of the box and will not be populated if present on your model.
The code for the State template looks like this:

[SitecoreType(TemplateId = "{00000000-0000-0000-0000-000000000002}", AutoMap = true)]
public class State
{
    public virtual Guid Id { get; set; }
    public virtual string Name { get; set; }
    public virtual string Abbreviation { get; set; }
}

Mapping from Sitecore

Mapping from Sitecore with concrete models is no different than mapping from Sitecore with interface models. See the examples above for how to map list data from Sitecore to concrete models.

Modifying List Fields

Populating list fields with concrete models is slightly different than with interface models. With concrete models we are able to instantiate our own objects as shown below; we don’t have to rely on the SitecoreContext or SitecoreService to instantiate new objects for us. Instantiate your object, initialize the list property, populate it with items from Sitecore, and use the SitecoreContext or SitecoreService to create your object in Sitecore.

using (var sitecoreContext = new SitecoreContext())
{
    var georgia = sitecoreContext.GetItem<State>("/sitecore/content/States/Georgia");
    var productsFolder = sitecoreContext.GetItem<Item>("/sitecore/content/Products");
    var product = new Product
    {
        Name = "New Concrete Product",
        EligibleStates = new List<State>()
    };
    product.EligibleStates.Add(georgia);
    sitecoreContext.Create(productsFolder, product);
}

One thing to note when working with concrete models is that you can only add items that already exist in Sitecore to lists; that is, you cannot instantiate a new item, add it to a list, and expect Glass Mapper to save it to Sitecore. This wasn’t an issue with interface models, where we had to rely on the SitecoreContext or SitecoreService to first create our objects (in Sitecore) before using them.
Consider the following example where I try to create a new state, California, and add it to the eligible states for my new product:

using (var sitecoreContext = new SitecoreContext())
{
    var california = new State
    {
        Name = "California",
        Abbreviation = "CA"
    };
    var productsFolder = sitecoreContext.GetItem<Item>("/sitecore/content/Products");
    var product = new Product
    {
        Name = "New Concrete Product",
        EligibleStates = new List<State>()
    };
    product.EligibleStates.Add(california);
    sitecoreContext.Create(productsFolder, product);
}

Glass Mapper will throw an exception on sitecoreContext.Create because it doesn’t know what to do with the new state that was created, similar to this:

Could not find item to save value AbstractPropertyConfiguration Property: EligibleStates Type: Glass.Mapper.FakePropertyInfo

Although Glass Mapper doesn’t support this out of the box, the brilliant Mike Skutta has implemented a task that you can plug into Glass Mapper to get this functionality.

Closing Thoughts

Glass Mapper does an excellent job of abstracting away the Sitecore API behind standard .NET classes. Do you make use of Glass Mapper for list fields in your projects? Let me know how you use it in the comments.

Leave a Reply

Your email address will not be published. Required fields are marked *

This site uses Akismet to reduce spam. Learn how your comment data is processed.

Corey Smith, Sitecore MVP & Lead Technical Consultant

I'm a Sitecore MVP and Lead Technical Consultant at Perficient focusing on Sitecore and custom .NET development.

More from this Author

Categories
Follow Us