Building an ASP.NET MVC Master Page Menu Dynamically, Based on the current User's "Role"

35,541

Solution 1

I have done something like this:

  • use a common base class for my controllers ('layer supertype')
  • in the BaseController, override OnActionExecuted (you could also define an ActionFilter attribute for this)

Something like this:

    protected override void OnActionExecuted(ActionExecutedContext filterContext)
    {
        // build list of menu items based on user's permissions, and add it to ViewData
        IEnumerable<MenuItem> menu = BuildMenu(); 
        ViewData["Menu"] = menu;
    }

In the master page:

    <% var model = ViewData["Menu"] as IEnumerable<MenuItem>; %>
    <% Html.RenderPartial("Menu", model); %>

(Note: in reality, I have a MasterViewModel that contains among others the menu model)

Solution 2

Did you hear about MvcContrib's MenuBuilder?

If not, I suggest you take a look at it. The example project UI is a good way to start learning how to use it.

Solution 3

Nobody's mentioned MvcSiteMapProvider which does this and can be easily integrated into your Visual Studio project using NuGet.

Solution 4

Usually I just check the role in a similar way you did and then either render a partial view with the links or just create them. Something like this using Razor syntax. I use T4MVC for actions.

@if(User.IsInRole("Admin"))
{
    <li><a href="@Url.Action(MVC.Admin.User.Index())">Users</a></li>
}

For security I use Fluent Security. Hope this helps.

Share:
35,541
Nate
Author by

Nate

Dad. Geek. Gamer. Web developer. Cloud user. Old Car enthusiast. Blogger.

Updated on January 30, 2020

Comments

  • Nate
    Nate over 4 years

    I've seen some similar questions, but none that look like what I'm trying to do.

    This is my current implementation w/out any security:

    <div id="menucontainer">
        <ul id="menu">              
            <li><%= Html.ActionLink("Main List", "Index", "AController")%></li>
            <li><%= Html.ActionLink("Product List", "Index", "BController")%></li>
            <li><%= Html.ActionLink("Company List", "Index", "CController")%></li>
            <li><%= Html.ActionLink("User List", "Index", "DController")%></li>
        </ul>
    </div>
    

    This is fine, and the above works. I have [Authorize] Attributes setup on the Actions for CController and DController to prevent unauthorized access -- but I'd like to remove those items from the menu for users who don't have the correct Role, because when they see it and click on it and it tells them they don't have permission, they'll want it. If they don't know it's there, that's just better for everyone involved...

    Something like this is ultimately the goal I'm trying to get at, but I'm looking for the more MVC Flavored aproach, where the "view" is "dumb":

    <div id="menucontainer">
        <ul id="menu">              
            <li><%= Html.ActionLink("Main List", "Index", "AController")%></li>
            <li><%= Html.ActionLink("Product List", "Index", "BController")%></li>
            <% If(Role = Roles.Admin) { %>
            <li><%= Html.ActionLink("Company List", "Index", "CController")%></li>
            <li><%= Html.ActionLink("User List", "Index", "DController")%></li>
            <% } %>
        </ul>
    </div>