2018-07-18 15:42:12 +10:00
|
|
|
|
using System;
|
|
|
|
|
|
using System.Collections.Generic;
|
2017-07-19 13:42:47 +02:00
|
|
|
|
using System.Linq;
|
2018-03-27 10:04:07 +02:00
|
|
|
|
using System.Web.Security;
|
|
|
|
|
|
using AutoMapper;
|
2017-07-19 13:42:47 +02:00
|
|
|
|
using Umbraco.Core;
|
2018-03-27 10:04:07 +02:00
|
|
|
|
using Umbraco.Core.Composing;
|
2017-07-19 13:42:47 +02:00
|
|
|
|
using Umbraco.Core.Models;
|
2018-03-27 10:04:07 +02:00
|
|
|
|
using Umbraco.Core.Models.Membership;
|
2017-07-19 13:42:47 +02:00
|
|
|
|
using Umbraco.Core.Security;
|
|
|
|
|
|
using Umbraco.Core.Services;
|
|
|
|
|
|
using Umbraco.Web.Models.ContentEditing;
|
|
|
|
|
|
|
|
|
|
|
|
namespace Umbraco.Web.Models.Mapping
|
|
|
|
|
|
{
|
|
|
|
|
|
/// <summary>
|
2019-01-26 10:52:19 -05:00
|
|
|
|
/// A custom tab/property resolver for members which will ensure that the built-in membership properties are or aren't displayed
|
2017-07-19 13:42:47 +02:00
|
|
|
|
/// depending on if the member type has these properties
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <remarks>
|
|
|
|
|
|
/// This also ensures that the IsLocked out property is readonly when the member is not locked out - this is because
|
|
|
|
|
|
/// an admin cannot actually set isLockedOut = true, they can only unlock.
|
|
|
|
|
|
/// </remarks>
|
2018-03-27 10:04:07 +02:00
|
|
|
|
internal class MemberTabsAndPropertiesResolver : TabsAndPropertiesResolver<IMember, MemberDisplay>
|
2017-07-19 13:42:47 +02:00
|
|
|
|
{
|
2018-07-18 14:34:32 +10:00
|
|
|
|
private readonly IUmbracoContextAccessor _umbracoContextAccessor;
|
2017-07-19 13:42:47 +02:00
|
|
|
|
private readonly ILocalizedTextService _localizedTextService;
|
2019-02-05 14:06:48 +01:00
|
|
|
|
private readonly IMemberTypeService _memberTypeService;
|
2018-03-27 10:04:07 +02:00
|
|
|
|
private readonly IMemberService _memberService;
|
|
|
|
|
|
private readonly IUserService _userService;
|
2017-07-19 13:42:47 +02:00
|
|
|
|
|
2019-02-07 14:01:00 +11:00
|
|
|
|
public MemberTabsAndPropertiesResolver(IUmbracoContextAccessor umbracoContextAccessor, ILocalizedTextService localizedTextService, IMemberService memberService, IUserService userService, IMemberTypeService memberTypeService)
|
|
|
|
|
|
: base(localizedTextService)
|
2017-07-19 13:42:47 +02:00
|
|
|
|
{
|
2019-02-05 14:06:48 +01:00
|
|
|
|
_umbracoContextAccessor = umbracoContextAccessor ?? throw new ArgumentNullException(nameof(umbracoContextAccessor));
|
|
|
|
|
|
_localizedTextService = localizedTextService ?? throw new ArgumentNullException(nameof(localizedTextService));
|
|
|
|
|
|
_memberService = memberService ?? throw new ArgumentNullException(nameof(memberService));
|
|
|
|
|
|
_userService = userService ?? throw new ArgumentNullException(nameof(userService));
|
|
|
|
|
|
_memberTypeService = memberTypeService ?? throw new ArgumentNullException(nameof(memberTypeService));
|
2017-07-19 13:42:47 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2018-03-27 10:04:07 +02:00
|
|
|
|
/// <inheritdoc />
|
2019-01-26 10:52:19 -05:00
|
|
|
|
/// <remarks>Overridden to deal with custom member properties and permissions.</remarks>
|
2018-03-27 10:04:07 +02:00
|
|
|
|
public override IEnumerable<Tab<ContentPropertyDisplay>> Resolve(IMember source, MemberDisplay destination, IEnumerable<Tab<ContentPropertyDisplay>> destMember, ResolutionContext context)
|
2017-07-19 13:42:47 +02:00
|
|
|
|
{
|
|
|
|
|
|
var provider = Core.Security.MembershipProviderExtensions.GetMembersMembershipProvider();
|
|
|
|
|
|
|
2019-02-05 14:06:48 +01:00
|
|
|
|
var memberType = _memberTypeService.Get(source.ContentTypeId);
|
|
|
|
|
|
|
|
|
|
|
|
IgnoreProperties = memberType.CompositionPropertyTypes
|
2017-07-19 13:42:47 +02:00
|
|
|
|
.Where(x => x.HasIdentity == false)
|
|
|
|
|
|
.Select(x => x.Alias)
|
|
|
|
|
|
.ToArray();
|
|
|
|
|
|
|
2018-03-27 10:04:07 +02:00
|
|
|
|
var resolved = base.Resolve(source, destination, destMember, context);
|
2017-07-19 13:42:47 +02:00
|
|
|
|
|
|
|
|
|
|
if (provider.IsUmbracoMembershipProvider() == false)
|
|
|
|
|
|
{
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// it's a generic provider so update the locked out property based on our known constant alias
|
2018-03-27 10:04:07 +02:00
|
|
|
|
var isLockedOutProperty = resolved.SelectMany(x => x.Properties).FirstOrDefault(x => x.Alias == Constants.Conventions.Member.IsLockedOut);
|
|
|
|
|
|
if (isLockedOutProperty?.Value != null && isLockedOutProperty.Value.ToString() != "1")
|
2017-07-19 13:42:47 +02:00
|
|
|
|
{
|
|
|
|
|
|
isLockedOutProperty.View = "readonlyvalue";
|
|
|
|
|
|
isLockedOutProperty.Value = _localizedTextService.Localize("general/no");
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
2018-10-23 16:12:23 +11:00
|
|
|
|
var umbracoProvider = (IUmbracoMemberTypeMembershipProvider)provider;
|
2017-07-19 13:42:47 +02:00
|
|
|
|
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// This is kind of a hack because a developer is supposed to be allowed to set their property editor - would have been much easier
|
|
|
|
|
|
// if we just had all of the membership provider fields on the member table :(
|
2017-07-19 13:42:47 +02:00
|
|
|
|
// TODO: But is there a way to map the IMember.IsLockedOut to the property ? i dunno.
|
2018-03-27 10:04:07 +02:00
|
|
|
|
var isLockedOutProperty = resolved.SelectMany(x => x.Properties).FirstOrDefault(x => x.Alias == umbracoProvider.LockPropertyTypeAlias);
|
|
|
|
|
|
if (isLockedOutProperty?.Value != null && isLockedOutProperty.Value.ToString() != "1")
|
2017-07-19 13:42:47 +02:00
|
|
|
|
{
|
|
|
|
|
|
isLockedOutProperty.View = "readonlyvalue";
|
|
|
|
|
|
isLockedOutProperty.Value = _localizedTextService.Localize("general/no");
|
|
|
|
|
|
}
|
2018-03-27 10:04:07 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2018-07-18 14:34:32 +10:00
|
|
|
|
var umbracoContext = _umbracoContextAccessor.UmbracoContext;
|
2018-03-27 10:04:07 +02:00
|
|
|
|
if (umbracoContext != null
|
|
|
|
|
|
&& umbracoContext.Security.CurrentUser != null
|
|
|
|
|
|
&& umbracoContext.Security.CurrentUser.AllowedSections.Any(x => x.Equals(Constants.Applications.Settings)))
|
|
|
|
|
|
{
|
|
|
|
|
|
var memberTypeLink = string.Format("#/member/memberTypes/edit/{0}", source.ContentTypeId);
|
2017-07-19 13:42:47 +02:00
|
|
|
|
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// Replace the doctype property
|
2018-03-27 10:04:07 +02:00
|
|
|
|
var docTypeProperty = resolved.SelectMany(x => x.Properties)
|
|
|
|
|
|
.First(x => x.Alias == string.Format("{0}doctype", Constants.PropertyEditors.InternalGenericPropertiesPrefix));
|
|
|
|
|
|
docTypeProperty.Value = new List<object>
|
|
|
|
|
|
{
|
|
|
|
|
|
new
|
|
|
|
|
|
{
|
|
|
|
|
|
linkText = source.ContentType.Name,
|
|
|
|
|
|
url = memberTypeLink,
|
|
|
|
|
|
target = "_self",
|
|
|
|
|
|
icon = "icon-item-arrangement"
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
docTypeProperty.View = "urllist";
|
2017-07-19 13:42:47 +02:00
|
|
|
|
}
|
2018-03-27 10:04:07 +02:00
|
|
|
|
|
|
|
|
|
|
return resolved;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
protected override IEnumerable<ContentPropertyDisplay> GetCustomGenericProperties(IContentBase content)
|
|
|
|
|
|
{
|
2018-10-23 16:12:23 +11:00
|
|
|
|
var member = (IMember)content;
|
2018-03-27 10:04:07 +02:00
|
|
|
|
var membersProvider = Core.Security.MembershipProviderExtensions.GetMembersMembershipProvider();
|
|
|
|
|
|
|
|
|
|
|
|
var genericProperties = new List<ContentPropertyDisplay>
|
|
|
|
|
|
{
|
2018-10-23 16:12:23 +11:00
|
|
|
|
new ContentPropertyDisplay
|
|
|
|
|
|
{
|
2018-10-23 19:25:48 +11:00
|
|
|
|
Alias = $"{Constants.PropertyEditors.InternalGenericPropertiesPrefix}id",
|
2018-10-23 16:12:23 +11:00
|
|
|
|
Label = _localizedTextService.Localize("general/id"),
|
|
|
|
|
|
Value = new List<string> {member.Id.ToString(), member.Key.ToString()},
|
|
|
|
|
|
View = "idwithguid"
|
|
|
|
|
|
},
|
2018-03-27 10:04:07 +02:00
|
|
|
|
new ContentPropertyDisplay
|
|
|
|
|
|
{
|
|
|
|
|
|
Alias = $"{Constants.PropertyEditors.InternalGenericPropertiesPrefix}doctype",
|
|
|
|
|
|
Label = _localizedTextService.Localize("content/membertype"),
|
|
|
|
|
|
Value = _localizedTextService.UmbracoDictionaryTranslate(member.ContentType.Name),
|
2019-02-15 11:42:29 +01:00
|
|
|
|
View = Current.PropertyEditors[Constants.PropertyEditors.Aliases.Label].GetValueEditor().View
|
2018-03-27 10:04:07 +02:00
|
|
|
|
},
|
|
|
|
|
|
GetLoginProperty(_memberService, member, _localizedTextService),
|
|
|
|
|
|
new ContentPropertyDisplay
|
|
|
|
|
|
{
|
|
|
|
|
|
Alias = $"{Constants.PropertyEditors.InternalGenericPropertiesPrefix}email",
|
|
|
|
|
|
Label = _localizedTextService.Localize("general/email"),
|
|
|
|
|
|
Value = member.Email,
|
|
|
|
|
|
View = "email",
|
|
|
|
|
|
Validation = {Mandatory = true}
|
|
|
|
|
|
},
|
|
|
|
|
|
new ContentPropertyDisplay
|
|
|
|
|
|
{
|
|
|
|
|
|
Alias = $"{Constants.PropertyEditors.InternalGenericPropertiesPrefix}password",
|
|
|
|
|
|
Label = _localizedTextService.Localize("password"),
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// NOTE: The value here is a json value - but the only property we care about is the generatedPassword one if it exists, the newPassword exists
|
2018-03-27 10:04:07 +02:00
|
|
|
|
// only when creating a new member and we want to have a generated password pre-filled.
|
|
|
|
|
|
Value = new Dictionary<string, object>
|
|
|
|
|
|
{
|
2019-01-27 01:17:32 -05:00
|
|
|
|
// TODO: why ignoreCase, what are we doing here?!
|
2018-03-27 10:04:07 +02:00
|
|
|
|
{"generatedPassword", member.GetAdditionalDataValueIgnoreCase("GeneratedPassword", null)},
|
|
|
|
|
|
{"newPassword", member.GetAdditionalDataValueIgnoreCase("NewPassword", null)},
|
|
|
|
|
|
},
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// TODO: Hard coding this because the changepassword doesn't necessarily need to be a resolvable (real) property editor
|
2018-03-27 10:04:07 +02:00
|
|
|
|
View = "changepassword",
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// initialize the dictionary with the configuration from the default membership provider
|
2018-03-27 10:04:07 +02:00
|
|
|
|
Config = new Dictionary<string, object>(membersProvider.GetConfiguration(_userService))
|
|
|
|
|
|
{
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// the password change toggle will only be displayed if there is already a password assigned.
|
2018-03-27 10:04:07 +02:00
|
|
|
|
{"hasPassword", member.RawPasswordValue.IsNullOrWhiteSpace() == false}
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
new ContentPropertyDisplay
|
|
|
|
|
|
{
|
|
|
|
|
|
Alias = $"{Constants.PropertyEditors.InternalGenericPropertiesPrefix}membergroup",
|
|
|
|
|
|
Label = _localizedTextService.Localize("content/membergroup"),
|
|
|
|
|
|
Value = GetMemberGroupValue(member.Username),
|
|
|
|
|
|
View = "membergroups",
|
|
|
|
|
|
Config = new Dictionary<string, object> {{"IsRequired", true}}
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
return genericProperties;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Overridden to assign the IsSensitive property values
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="content"></param>
|
|
|
|
|
|
/// <param name="properties"></param>
|
2018-04-04 01:59:51 +10:00
|
|
|
|
/// <param name="context"></param>
|
2018-03-27 10:04:07 +02:00
|
|
|
|
/// <returns></returns>
|
2018-07-18 15:42:12 +10:00
|
|
|
|
protected override List<ContentPropertyDisplay> MapProperties(IContentBase content, List<Property> properties, ResolutionContext context)
|
2018-03-27 10:04:07 +02:00
|
|
|
|
{
|
2018-07-18 15:42:12 +10:00
|
|
|
|
var result = base.MapProperties(content, properties, context);
|
2018-03-27 10:04:07 +02:00
|
|
|
|
var member = (IMember)content;
|
2019-02-05 14:06:48 +01:00
|
|
|
|
var memberType = _memberTypeService.Get(member.ContentTypeId);
|
2018-03-27 10:04:07 +02:00
|
|
|
|
|
2018-07-18 15:42:12 +10:00
|
|
|
|
var umbracoContext = _umbracoContextAccessor.UmbracoContext;
|
|
|
|
|
|
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// now update the IsSensitive value
|
2018-03-27 10:04:07 +02:00
|
|
|
|
foreach (var prop in result)
|
|
|
|
|
|
{
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// check if this property is flagged as sensitive
|
2018-03-27 10:04:07 +02:00
|
|
|
|
var isSensitiveProperty = memberType.IsSensitiveProperty(prop.Alias);
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// check permissions for viewing sensitive data
|
2018-07-18 15:42:12 +10:00
|
|
|
|
if (isSensitiveProperty && (umbracoContext == null || umbracoContext.Security.CurrentUser.HasAccessToSensitiveData() == false))
|
2018-03-27 10:04:07 +02:00
|
|
|
|
{
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// mark this property as sensitive
|
2018-03-27 10:04:07 +02:00
|
|
|
|
prop.IsSensitive = true;
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// mark this property as readonly so that it does not post any data
|
2018-03-27 10:04:07 +02:00
|
|
|
|
prop.Readonly = true;
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// replace this editor with a sensitive value
|
2018-03-27 10:04:07 +02:00
|
|
|
|
prop.View = "sensitivevalue";
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// clear the value
|
2018-03-27 10:04:07 +02:00
|
|
|
|
prop.Value = null;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Returns the login property display field
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="memberService"></param>
|
|
|
|
|
|
/// <param name="member"></param>
|
|
|
|
|
|
/// <param name="display"></param>
|
|
|
|
|
|
/// <param name="localizedText"></param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
/// <remarks>
|
|
|
|
|
|
/// If the membership provider installed is the umbraco membership provider, then we will allow changing the username, however if
|
2019-01-26 10:52:19 -05:00
|
|
|
|
/// the membership provider is a custom one, we cannot allow changing the username because MembershipProvider's do not actually natively
|
2018-03-27 10:04:07 +02:00
|
|
|
|
/// allow that.
|
|
|
|
|
|
/// </remarks>
|
|
|
|
|
|
internal static ContentPropertyDisplay GetLoginProperty(IMemberService memberService, IMember member, ILocalizedTextService localizedText)
|
|
|
|
|
|
{
|
|
|
|
|
|
var prop = new ContentPropertyDisplay
|
|
|
|
|
|
{
|
|
|
|
|
|
Alias = $"{Constants.PropertyEditors.InternalGenericPropertiesPrefix}login",
|
|
|
|
|
|
Label = localizedText.Localize("login"),
|
|
|
|
|
|
Value = member.Username
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
var scenario = memberService.GetMembershipScenario();
|
|
|
|
|
|
|
2019-01-26 10:52:19 -05:00
|
|
|
|
// only allow editing if this is a new member, or if the membership provider is the Umbraco one
|
2018-03-27 10:04:07 +02:00
|
|
|
|
if (member.HasIdentity == false || scenario == MembershipScenario.NativeUmbraco)
|
|
|
|
|
|
{
|
|
|
|
|
|
prop.View = "textbox";
|
|
|
|
|
|
prop.Validation.Mandatory = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
prop.View = "readonlyvalue";
|
|
|
|
|
|
}
|
|
|
|
|
|
return prop;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
internal static IDictionary<string, bool> GetMemberGroupValue(string username)
|
|
|
|
|
|
{
|
|
|
|
|
|
var userRoles = username.IsNullOrWhiteSpace() ? null : Roles.GetRolesForUser(username);
|
|
|
|
|
|
|
|
|
|
|
|
// create a dictionary of all roles (except internal roles) + "false"
|
|
|
|
|
|
var result = Roles.GetAllRoles().Distinct()
|
|
|
|
|
|
// if a role starts with __umbracoRole we won't show it as it's an internal role used for public access
|
|
|
|
|
|
.Where(x => x.StartsWith(Constants.Conventions.Member.InternalRolePrefix) == false)
|
|
|
|
|
|
.ToDictionary(x => x, x => false);
|
|
|
|
|
|
|
|
|
|
|
|
// if user has no roles, just return the dictionary
|
|
|
|
|
|
if (userRoles == null) return result;
|
|
|
|
|
|
|
|
|
|
|
|
// else update the dictionary to "true" for the user roles (except internal roles)
|
|
|
|
|
|
foreach (var userRole in userRoles.Where(x => x.StartsWith(Constants.Conventions.Member.InternalRolePrefix) == false))
|
|
|
|
|
|
result[userRole] = true;
|
|
|
|
|
|
|
|
|
|
|
|
return result;
|
2017-07-19 13:42:47 +02:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2017-07-20 11:21:28 +02:00
|
|
|
|
}
|