7

在我的MVC3项目中,我安装了 Maartenba 的MvcSiteMapProvider v.3.2.1,并且我创建了一个非常简单的静态两级菜单。下面是概念图结构。

- Home
- Member Center
    - Member Listing [SELECTED]
    - Event Calendar
    - Documents
- Administration

现在,成员列表下有许多子页面(例如,详细信息、编辑等),但我希望这些显示为 3 级菜单项(主要是因为它们与特定的成员 ID 相关联)。但是,我确实希望所有这些第三级页面都“绑定”到成员列表菜单节点,以便在这些页面上显示为选中状态。

我的 Mvc.SiteMap 文件中有以下代码:

<mvcSiteMapNode title="Home" controller="Home" action="Index">
  <mvcSiteMapNode title="Member Center" area="Members" controller="Home" action="Index" roles="Approved Member" >
    <mvcSiteMapNode title="Member Listing" area="Members" controller="Member" action="List" />
    <mvcSiteMapNode title="Event Calendar" area="Members" controller="Event" action="List" />
    <mvcSiteMapNode title="Documents" area="Members" controller="Document" action="List" />
  </mvcSiteMapNode>
  <mvcSiteMapNode title="Administration" area="Admin" controller="Home" action="Index" roles="Site Administrator" >
  </mvcSiteMapNode>
</mvcSiteMapNode> 

为了呈现菜单,我在 _Layout.cshtml 文件中使用了以下代码:

@Html.MvcSiteMap().Menu(1, true, true, 1, true, true)

最后,我修改了 SiteMapNodeModel.cshtml 文件,以便将“selectedMenuItem”类添加到与用户正在查看的页面相关的节点。这是呈现菜单节点的代码片段。

@model SiteMapNodeModel
  <a href="@Model.Url" class="@(Model.IsCurrentNode ? "selectedMenuItem" : "")">@Model.Title</a>

地图的显示和导航工作得很好,直到我进一步导航到成员区域。例如,如果我经过Members/Member/List(正确显示菜单)并进入类似的页面Members/Member/Detail/1,则会员中心下的子节点(“会员列表”、“活动日历”等)消失。因此,这是我在当前代码中遇到的两个问题:

  1. 我想指定任何给定页面是“会员中心”父菜单节点的一部分,这样“会员中心”的子菜单节点都会显示出来,而不管给定页面是否定义为菜单中的特定节点结构体。

  2. 我想指定(可能在视图或控制器操作中)特定页面应绑定到特定菜单节点。例如,当用户位于 时Members/Member/Detail/1,我只想指定“成员列表”子节点,IsCurrentNode以便 SiteMapNodeModel.cshtml 文件使用“selectedMenuItem”类正确装饰它。

有什么建议么?

4

3 回答 3

6

您可以将 3 级节点添加到站点地图 XML 并指定可见性以将它们从菜单中隐藏。这是仅在面包屑中显示的节点声明:

<mvcSiteMapNode area="Members"
                controller="Member"
                action="Detail"
                visibility="SiteMapPathHelper,!*"
                title="Member details" />

编辑:

据我所知,你不能设置 IsCurrentNode。但是您可以使用以下代码检查当前是否选择了菜单节点(我在 SiteMapNodeModel 显示模板中使用它):

IList<string> classes = new List<string> ();
if (Model.IsCurrentNode || Model.IsInCurrentPath && !Model.Children.Any ())
{
    classes.Add ("menu-current");
}
于 2012-04-11T19:20:08.770 回答
1

继 Max Kiselev 的回答之后,如果您想使用该技术但能够在控制器操作上使用属性,我执行了以下操作:

定义自定义可见性提供程序:

public class AlwaysInvisibleVisibilityProvider : ISiteMapNodeVisibilityProvider
{
    public bool IsVisible(SiteMapNode node, HttpContext context, IDictionary<string, object> sourceMetadata)
    {
        return false;
    }
}

然后子类化 MvcSiteMapNodeAttribute:

public class InvisibleMvcSiteMapNodeAttribute : MvcSiteMapNodeAttribute
{
    public InvisibleMvcSiteMapNodeAttribute(string key, string parentKey)
    {
        Key = key;
        ParentKey = parentKey;
        VisibilityProvider = typeof (AlwaysInvisibleVisibilityProvider).AssemblyQualifiedName;
    }
}

然后您可以在控制器操作中使用它:

[HttpGet]
[InvisibleMvcSiteMapNodeAttribute("ThisNodeKey", "ParentNodeKey")] 
public ViewResult OrderTimeout()
{
    return View("Timeout");
}
于 2012-05-18T08:22:50.403 回答
1

添加到 Max 的答案中,我还将为 SiteMapNodeModel 创建一个扩展方法。您可以使用它来实现执行此操作所需的所有自定义逻辑:

public static class SiteMapNodeModelExtender
{
  public static bool IsRealCurrentNode(this SiteMapNodeModel node)
  {
    // Logic to determine the "real" current node...
    // A naive implementation could be:
    var currentPath = HttpContext.Current.Request.Url.AbsolutePath;
    return currentPath.StartsWith("Members/Member/") && node.Title.Equals("Member Center")
  }
}

并相应地更改显示模板:

/* Also check IsRealCurrentNode, depending on the use case maybe only
IsRealCurrentNode */
@if ((Model.IsCurrentNode || Model.IsRealCurrentNode()) && Model.SourceMetadata["HtmlHelper"].ToString() != "MvcSiteMapProvider.Web.Html.MenuHelper")  { 
  <text>@Model.Title</text>
} else if (Model.IsClickable) { 
  <a href="@Model.Url ">@Model.Title</a>
} else { 
  <text>@Model.Title</text>
}
于 2012-04-12T09:15:12.523 回答