Files
Umbraco-CMS/src/Umbraco.Web/Models/Mapping/EntityMapperProfile.cs

217 lines
12 KiB
C#
Raw Normal View History

2017-07-19 13:42:47 +02:00
using System;
using System.Collections.Generic;
using System.Linq;
using AutoMapper;
using Examine;
using Examine.LuceneEngine.Providers;
using Examine.LuceneEngine;
using Umbraco.Core;
using Umbraco.Core.Models;
using Umbraco.Core.Models.Entities;
2017-07-19 13:42:47 +02:00
using Umbraco.Core.Models.Membership;
using Umbraco.Web.Models.ContentEditing;
using Umbraco.Examine;
2017-07-19 13:42:47 +02:00
namespace Umbraco.Web.Models.Mapping
{
2017-09-20 20:06:46 +02:00
internal class EntityMapperProfile : Profile
2017-07-19 13:42:47 +02:00
{
private static string GetContentTypeIcon(IEntitySlim entity)
=> entity is ContentEntitySlim contentEntity ? contentEntity.ContentTypeIcon : null;
2017-09-20 20:06:46 +02:00
public EntityMapperProfile()
2017-07-19 13:42:47 +02:00
{
// create, capture, cache
var contentTypeUdiResolver = new ContentTypeUdiResolver();
CreateMap<IEntitySlim, EntityBasic>()
2019-02-06 21:14:50 +11:00
.ForMember(dest => dest.Name, opt => opt.MapFrom<NameResolver>())
.ForMember(dest => dest.Udi, opt => opt.MapFrom(src => Udi.Create(ObjectTypes.GetUdiType(src.NodeObjectType), src.Key)))
.ForMember(dest => dest.Icon, opt => opt.MapFrom(src => GetContentTypeIcon(src)))
2018-03-27 10:04:07 +02:00
.ForMember(dest => dest.Trashed, opt => opt.MapFrom(src => src.Trashed))
2017-07-19 13:42:47 +02:00
.ForMember(dest => dest.Alias, opt => opt.Ignore())
.AfterMap((src, dest) =>
{
if (src.NodeObjectType == Constants.ObjectTypes.Member && dest.Icon.IsNullOrWhiteSpace())
2017-07-19 13:42:47 +02:00
{
dest.Icon = "icon-user";
}
dest.AdditionalData.Add("IsContainer", src.IsContainer);
2017-07-19 13:42:47 +02:00
});
CreateMap<PropertyType, EntityBasic>()
.ForMember(dest => dest.Udi, opt => opt.Ignore())
2019-02-06 21:14:50 +11:00
.ForMember(dest => dest.Icon, opt => opt.MapFrom(_ => "icon-box"))
.ForMember(dest => dest.Path, opt => opt.MapFrom(_ => ""))
.ForMember(dest => dest.ParentId, opt => opt.MapFrom(_ => -1))
2017-07-19 13:42:47 +02:00
.ForMember(dest => dest.Trashed, opt => opt.Ignore())
.ForMember(dest => dest.AdditionalData, opt => opt.Ignore());
CreateMap<PropertyGroup, EntityBasic>()
.ForMember(dest => dest.Udi, opt => opt.Ignore())
2019-02-06 21:14:50 +11:00
.ForMember(dest => dest.Icon, opt => opt.MapFrom(_ => "icon-tab"))
.ForMember(dest => dest.Path, opt => opt.MapFrom(_ => ""))
.ForMember(dest => dest.ParentId, opt => opt.MapFrom(_ => -1))
2017-07-19 13:42:47 +02:00
//in v6 the 'alias' is it's lower cased name so we'll stick to that.
.ForMember(dest => dest.Alias, opt => opt.MapFrom(src => src.Name.ToLowerInvariant()))
.ForMember(dest => dest.Trashed, opt => opt.Ignore())
.ForMember(dest => dest.AdditionalData, opt => opt.Ignore());
CreateMap<IUser, EntityBasic>()
.ForMember(dest => dest.Udi, opt => opt.Ignore())
2019-02-06 21:14:50 +11:00
.ForMember(dest => dest.Icon, opt => opt.MapFrom(_ => "icon-user"))
.ForMember(dest => dest.Path, opt => opt.MapFrom(_ => ""))
.ForMember(dest => dest.ParentId, opt => opt.MapFrom(_ => -1))
2017-07-19 13:42:47 +02:00
.ForMember(dest => dest.Alias, opt => opt.MapFrom(src => src.Username))
.ForMember(dest => dest.Trashed, opt => opt.Ignore())
.ForMember(dest => dest.AdditionalData, opt => opt.Ignore());
CreateMap<ITemplate, EntityBasic>()
.ForMember(dest => dest.Udi, opt => opt.MapFrom(src => Udi.Create(Constants.UdiEntityType.Template, src.Key)))
2019-02-06 21:14:50 +11:00
.ForMember(dest => dest.Icon, opt => opt.MapFrom(_ => "icon-layout"))
2017-07-19 13:42:47 +02:00
.ForMember(dest => dest.Path, opt => opt.MapFrom(src => src.Path))
2019-02-06 21:14:50 +11:00
.ForMember(dest => dest.ParentId, opt => opt.MapFrom(_ => -1))
2017-07-19 13:42:47 +02:00
.ForMember(dest => dest.Trashed, opt => opt.Ignore())
.ForMember(dest => dest.AdditionalData, opt => opt.Ignore());
CreateMap<EntityBasic, ContentTypeSort>()
.ForMember(dest => dest.Id, opt => opt.MapFrom(src => new Lazy<int>(() => Convert.ToInt32(src.Id))))
.ForMember(dest => dest.SortOrder, opt => opt.Ignore());
CreateMap<IContentTypeComposition, EntityBasic>()
2019-02-06 21:14:50 +11:00
.ForMember(dest => dest.Udi, opt => opt.MapFrom(src => contentTypeUdiResolver.Resolve(src)))
2017-07-19 13:42:47 +02:00
.ForMember(dest => dest.Path, opt => opt.MapFrom(src => src.Path))
.ForMember(dest => dest.ParentId, opt => opt.MapFrom(src => src.ParentId))
.ForMember(dest => dest.Trashed, opt => opt.Ignore())
.ForMember(dest => dest.AdditionalData, opt => opt.Ignore());
CreateMap<EntitySlim, SearchResultEntity>()
.ForMember(dest => dest.Udi, opt => opt.MapFrom(src => Udi.Create(ObjectTypes.GetUdiType(src.NodeObjectType), src.Key)))
.ForMember(dest => dest.Icon, opt => opt.MapFrom(src=> GetContentTypeIcon(src)))
2017-09-12 16:22:16 +02:00
.ForMember(dest => dest.Trashed, opt => opt.Ignore())
.ForMember(dest => dest.Alias, opt => opt.Ignore())
.ForMember(dest => dest.Score, opt => opt.Ignore())
.AfterMap((entity, basic) =>
{
if (basic.Icon.IsNullOrWhiteSpace())
{
if (entity.NodeObjectType == Constants.ObjectTypes.Member)
2017-09-12 16:22:16 +02:00
basic.Icon = "icon-user";
else if (entity.NodeObjectType == Constants.ObjectTypes.DataType)
2017-09-12 16:22:16 +02:00
basic.Icon = "icon-autofill";
else if (entity.NodeObjectType == Constants.ObjectTypes.DocumentType)
2017-09-12 16:22:16 +02:00
basic.Icon = "icon-item-arrangement";
else if (entity.NodeObjectType == Constants.ObjectTypes.MediaType)
2017-09-12 16:22:16 +02:00
basic.Icon = "icon-thumbnails";
else if (entity.NodeObjectType == Constants.ObjectTypes.TemplateType)
2017-09-12 16:22:16 +02:00
basic.Icon = "icon-newspaper-alt";
}
});
CreateMap<ISearchResult, SearchResultEntity>()
2017-07-19 13:42:47 +02:00
//default to document icon
2017-09-12 16:22:16 +02:00
.ForMember(dest => dest.Score, opt => opt.MapFrom(result => result.Score))
2017-07-19 13:42:47 +02:00
.ForMember(dest => dest.Udi, opt => opt.Ignore())
.ForMember(dest => dest.Icon, opt => opt.Ignore())
.ForMember(dest => dest.Id, opt => opt.MapFrom(src => src.Id))
.ForMember(dest => dest.Name, opt => opt.Ignore())
.ForMember(dest => dest.Key, opt => opt.Ignore())
.ForMember(dest => dest.ParentId, opt => opt.Ignore())
.ForMember(dest => dest.Alias, opt => opt.Ignore())
.ForMember(dest => dest.Path, opt => opt.Ignore())
.ForMember(dest => dest.Trashed, opt => opt.Ignore())
.ForMember(dest => dest.AdditionalData, opt => opt.Ignore())
.AfterMap((src, dest) =>
2017-07-20 11:21:28 +02:00
{
// TODO: Properly map this (not aftermap)
2018-12-11 17:54:10 +11:00
2017-07-19 13:42:47 +02:00
//get the icon if there is one
dest.Icon = src.Values.ContainsKey(UmbracoExamineIndex.IconFieldName)
? src.Values[UmbracoExamineIndex.IconFieldName]
2017-07-19 13:42:47 +02:00
: "icon-document";
2018-11-26 17:20:15 +11:00
dest.Name = src.Values.ContainsKey("nodeName") ? src.Values["nodeName"] : "[no name]";
if (src.Values.ContainsKey(UmbracoExamineIndex.NodeKeyFieldName))
2017-07-19 13:42:47 +02:00
{
Guid key;
if (Guid.TryParse(src.Values[UmbracoExamineIndex.NodeKeyFieldName], out key))
2017-07-19 13:42:47 +02:00
{
dest.Key = key;
//need to set the UDI
if (src.Values.ContainsKey(LuceneIndex.CategoryFieldName))
2017-07-19 13:42:47 +02:00
{
switch (src.Values[LuceneIndex.CategoryFieldName])
2017-07-19 13:42:47 +02:00
{
case IndexTypes.Member:
dest.Udi = new GuidUdi(Constants.UdiEntityType.Member, dest.Key);
break;
case IndexTypes.Content:
dest.Udi = new GuidUdi(Constants.UdiEntityType.Document, dest.Key);
break;
case IndexTypes.Media:
dest.Udi = new GuidUdi(Constants.UdiEntityType.Media, dest.Key);
break;
}
}
}
}
2018-11-26 17:20:15 +11:00
if (src.Values.ContainsKey("parentID"))
2017-07-19 13:42:47 +02:00
{
int parentId;
2018-11-26 17:20:15 +11:00
if (int.TryParse(src.Values["parentID"], out parentId))
2017-07-19 13:42:47 +02:00
{
dest.ParentId = parentId;
}
else
{
dest.ParentId = -1;
}
}
dest.Path = src.Values.ContainsKey(UmbracoExamineIndex.IndexPathFieldName) ? src.Values[UmbracoExamineIndex.IndexPathFieldName] : "";
2017-07-20 11:21:28 +02:00
if (src.Values.ContainsKey(LuceneIndex.ItemTypeFieldName))
2017-07-19 13:42:47 +02:00
{
dest.AdditionalData.Add("contentType", src.Values[LuceneIndex.ItemTypeFieldName]);
2017-07-19 13:42:47 +02:00
}
});
CreateMap<ISearchResults, IEnumerable<SearchResultEntity>>()
.ConvertUsing(results => results.Select(Mapper.Map<SearchResultEntity>).ToList());
2017-07-19 13:42:47 +02:00
CreateMap<IEnumerable<ISearchResult>, IEnumerable<SearchResultEntity>>()
.ConvertUsing(results => results.Select(Mapper.Map<SearchResultEntity>).ToList());
2017-07-19 13:42:47 +02:00
}
2018-09-26 16:27:34 +02:00
/// <summary>
/// Resolves the name for a content item/content variant
/// </summary>
private class NameResolver : IValueResolver<IEntitySlim, EntityBasic, string>
2018-09-26 16:27:34 +02:00
{
public string Resolve(IEntitySlim source, EntityBasic destination, string destMember, ResolutionContext context)
2018-09-26 16:27:34 +02:00
{
if (!(source is DocumentEntitySlim doc))
return source.Name;
// invariant = only 1 name
if (!doc.Variations.VariesByCulture()) return source.Name;
// variant = depends on culture
var culture = context.Options.GetCulture();
// if there's no culture here, the issue is somewhere else (UI, whatever) - throw!
if (culture == null)
//throw new InvalidOperationException("Missing culture in mapping options.");
// TODO: we should throw, but this is used in various places that won't set a culture yet
2018-09-26 16:27:34 +02:00
return source.Name;
// if we don't have a name for a culture, it means the culture is not available, and
// hey we should probably not be mapping it, but it's too late, return a fallback name
return doc.CultureNames.TryGetValue(culture, out var name) && !name.IsNullOrWhiteSpace() ? name : $"({source.Name})";
2018-09-26 16:27:34 +02:00
}
}
2017-07-19 13:42:47 +02:00
}
2017-07-20 11:21:28 +02:00
}