diff --git a/src/MvcSiteMapProvider/MvcSiteMapProvider/DefaultSiteMapProvider.cs b/src/MvcSiteMapProvider/MvcSiteMapProvider/DefaultSiteMapProvider.cs index 5ab257ea..8d6a1aa9 100644 --- a/src/MvcSiteMapProvider/MvcSiteMapProvider/DefaultSiteMapProvider.cs +++ b/src/MvcSiteMapProvider/MvcSiteMapProvider/DefaultSiteMapProvider.cs @@ -25,8 +25,16 @@ namespace MvcSiteMapProvider public class DefaultSiteMapProvider : StaticSiteMapProvider { + #region Attribute names + + protected const string AttributesToInheritKey = "attributesToInherit"; + + #endregion + #region Private + private readonly char[] configValueSeparators = new[] { ';', ',' }; + protected const string RootName = "mvcSiteMap"; protected const string NodeName = "mvcSiteMapNode"; protected readonly XNamespace ns = "http://mvcsitemap.codeplex.com/schemas/MvcSiteMap-File-3.0"; @@ -42,6 +50,8 @@ public class DefaultSiteMapProvider protected List excludeAssembliesForScan = new List(); protected List includeAssembliesForScan = new List(); protected List attributesToIgnore = new List(); + protected string[] attributesToInherit = new string[0]; + #endregion #region Properties @@ -296,6 +306,12 @@ public override void Initialize(string name, NameValueCollection attributes) } } + // Which attributes in the sitemap XML should be inherited from parent nodes? + if(!string.IsNullOrEmpty(attributes[AttributesToInheritKey])) + { + attributesToInherit = attributes[AttributesToInheritKey].Split(configValueSeparators); + } + // Is a node key generator given? if (!string.IsNullOrEmpty(attributes["nodeKeyGenerator"])) { @@ -1442,35 +1458,17 @@ protected MvcSiteMapNode GetSiteMapNodeFromXmlElement(XElement node, SiteMapNode siteMapNode.ResourceKey = implicitResourceKey; // Create a route data dictionary - IDictionary routeValues = new Dictionary(); - AttributesToRouteValues(node, siteMapNode, routeValues); + siteMapNode.RouteValues = new Dictionary(); + AttributesToRouteValues(node, siteMapNode, siteMapNode.RouteValues); - // Inherit area and controller from parent - if (parentMvcNode != null) - { - if (siteMapNode["area"] == null) - { - siteMapNode["area"] = parentMvcNode.Area; - routeValues.Add("area", parentMvcNode.Area); - } - if (node.GetAttributeValue("controller") == "") - { - siteMapNode["controller"] = parentMvcNode.Controller; - routeValues.Add("controller", parentMvcNode.Controller); - } - var action = "action"; - if (node.GetAttributeValue(action) == String.Empty) - { - siteMapNode[action] = parentMvcNode.Action; - routeValues.Add(action, parentMvcNode.Action); - } - } + // Load inherited attributes + LoadInheritedAttribute("area", node, siteMapNode, parentMvcNode, defaultValue: string.Empty, isRouteValue: true); + LoadInheritedAttribute("controller", node, siteMapNode, parentMvcNode, isRouteValue: true); + LoadInheritedAttribute("action", node, siteMapNode, parentMvcNode, isRouteValue: true); - // Add defaults for area - if (!routeValues.ContainsKey("area")) + foreach(var attributeName in attributesToInherit) { - siteMapNode["area"] = ""; - routeValues.Add("area", ""); + LoadInheritedAttribute(attributeName, node, siteMapNode, parentMvcNode); } // Add defaults for SiteMapNodeUrlResolver @@ -1497,13 +1495,10 @@ protected MvcSiteMapNode GetSiteMapNodeFromXmlElement(XElement node, SiteMapNode var item = inheritedRouteParameter.Trim(); if (parentMvcNode != null && parentMvcNode.RouteValues.ContainsKey(item)) { - routeValues[item] = parentMvcNode.RouteValues[item]; + siteMapNode.RouteValues[item] = parentMvcNode.RouteValues[item]; } } - // Add route values to sitemap node - siteMapNode.RouteValues = routeValues; - // Add node's route defaults var httpContext = new HttpContextWrapper(HttpContext.Current); RouteData routeData = siteMapNode.GetRouteData(httpContext); @@ -1791,6 +1786,33 @@ protected void DecodeExternalUrl(SiteMapNode node) node.Url = HttpContext.Current.Server.UrlDecode(node.Url); } + /// + /// Get the attribute value from current node or from the parent nodes + /// + /// Name of the attribute + /// Xml node + /// MvcSiteMapNode to set the attribute to. + /// Parent MvcSiteMapNode + /// Default attribute value + /// If this flag is set to "true", than the attribute value will also be added to routeValues + private void LoadInheritedAttribute(string attributeName, XElement xmlNode, MvcSiteMapNode siteMapNode, MvcSiteMapNode parentNode, string defaultValue = null, bool isRouteValue = false) + { + var value = xmlNode.GetAttributeValue(attributeName); + var currentNode = parentNode; + + while(currentNode != null && string.IsNullOrEmpty(value)) + { + value = currentNode[attributeName]; + currentNode = currentNode.ParentNode as MvcSiteMapNode; + } + + siteMapNode[attributeName] = value ?? defaultValue; + if(isRouteValue) + { + siteMapNode.RouteValues[attributeName] = value ?? defaultValue; + } + } + #endregion } }