Как написать динамические данные в макете страницы в MVC 3 Razor?

У меня есть проект MVC 3 С# с двигателем Razor. Каковы способы и, я думаю, лучшие практики для записи динамических данных в _Layout.cshtml?

Например, возможно, мне нужно отобразить имя пользователя в правом верхнем углу моего веб-сайта, и это имя исходит из Session, DB или любого другого, основанного на том, что пользователь вошел в систему.

Спасибо.

UPDATE: Я также ищу хорошую практику для передачи определенных данных в элемент макета. Например, если мне нужно отобразить определенный файл CSS в зависимости от учетных данных входа в систему.

(В приведенном выше примере я думал об использовании помощников Url.)

Ответы

Ответ 1

По умолчанию интернет-приложение, созданное visual studio, использует _LogOnPartial.cshtml, чтобы сделать именно это.

Значение имени пользователя задается в действии LogOn для HomeController

Код для _LogOnPartial.cshtml

@if(Request.IsAuthenticated) {
    <text>Welcome <strong>@User.Identity.Name</strong>!
    [ @Html.ActionLink("Log Off", "LogOff", "Account") ]</text>
}
else {
    @:[ @Html.ActionLink("Log On", "LogOn", "Account") ]
}

User.Identity является частью поставщика членства aspnet.

Код для _Layout.cshtml

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8" />
    <title>@ViewBag.Title</title>
    <link href="@Url.Content("~/Content/Site.css")" rel="stylesheet" type="text/css" />
    <script src="@Url.Content("~/Scripts/jquery-1.5.1.min.js")" type="text/javascript"></script>
    <script src="@Url.Content("~/Scripts/modernizr-1.7.min.js")" type="text/javascript"></script>
</head>
<body>
    <div class="page">
        <header>
            <div id="title">
                <h1>Test</h1>
            </div>
            <div id="logindisplay">
                @Html.Partial("_LogOnPartial")
            </div>
            <nav>
                <ul id="menu">
                </ul>
            </nav>
        </header>
        <section id="main">
            @RenderBody()
        </section>
        <footer>
        </footer>
    </div>
</body>
</html>

Код действия входа в учетную запись AccountController

[HttpPost]
        public ActionResult LogOn(LogOnModel model, string returnUrl)
        {
            if (ModelState.IsValid)
            {
                if (Membership.ValidateUser(model.UserName, model.Password))
                {
                    FormsAuthentication.SetAuthCookie(model.UserName, model.RememberMe);
                    if (Url.IsLocalUrl(returnUrl) && returnUrl.Length > 1 && returnUrl.StartsWith("/")
                        && !returnUrl.StartsWith("//") && !returnUrl.StartsWith("/\\"))
                    {
                        return Redirect(returnUrl);
                    }
                    else
                    {
                        return RedirectToAction("Index", "Home");
                    }
                }
                else
                {
                    ModelState.AddModelError("", "The user name or password provided is incorrect.");
                }
            }

            // If we got this far, something failed, redisplay form
            return View(model);
        }

Код для класса ApplicationViewPage

public abstract class ApplicationViewPage<T> : WebViewPage<T>
    {
        protected override void InitializePage()
        {
            SetViewBagDefaultProperties();
            base.InitializePage();
        }

        private void SetViewBagDefaultProperties()
        {
            ViewBag.LayoutModel = new LayoutModel(Request.ServerVariables["SERVER_NAME"]);
        }

    }

Вышеприведенный код позволяет мне иметь ViewBag.LayoutModel, который содержит экземпляр класса LayoutModel на каждой странице.

Вот код для моего класса LayoutModel

public class LayoutModel
    {
        public string LayoutFile { get; set; }
        public string IpsTop { get; set; }
        public string IpsBottom { get; set; }
        public string ProfileTop { get; set; }
        public string ProfileBottom { get; set; }

        public LayoutModel(string hostname)
        {
            switch (hostname.ToLower())
            {
                default:

                    LayoutFile = "~/Views/Shared/_BnlLayout.cshtml";
                    IpsBottom = "~/Template/_BnlIpsBottom.cshtml";
                    IpsTop = "~/Template/_BnlTop.cshtml";
                    ProfileTop = "~/Template/_BnlProfileTop.cshtml";
                    break;

                case "something.com":
                    LayoutFile = "~/Views/Shared/_Layout.cshtml";
                    IpsBottom = "~/Template/_somethingBottom.cshtml";
                    IpsTop = "~/Template/_somethingTop.cshtml";
                    ProfileTop = "~/Template/_somethingProfileTop.cshtml";
                    break;
            }
        }
    }

Вот код для представления

@{
    ViewBag.Title = "PageTitle";
    Layout = @ViewBag.LayoutModel.LayoutFile; 
}
@using (Html.BeginForm())
{
    <span class="error">@ViewBag.ErrorMessage</span>
    <input type="hidden" name="Referrer" id="Referrer" value="@ViewBag.Referrer" />
    html stuff here       
}

Подробнее см. следующий вопрос. Убедитесь, что вы изменили свой web.config, как описано там: Как установить свойства ViewBag для всех представлений без использования базового класса для контроллеров?

Ответ 2

В дополнение к ответу atbebtg, чтобы отобразить материал в голове, вы хотите использовать поддержку раздела Razor. Разделы называются фрагментами шаблонов HTML, которые могут быть определены в представлениях и отображены в макете, где макет сочтет нужным. В макете вы вызываете @RenderSection("wellKnownSectionName") и в представлении, использующем макет, вы объявляете @section wellKnownSectionName { <link rel="stylesheet" href="@UserStylesheetUrl" /><script type="text/javascript" src="@UserScriptUrl"> }. Обычно вы хотите описать намерение раздела в его имени, например "documentHead".

Обновление: Если вы производите один и тот же шаблонный HTML на каждом представлении, он вместо этого перейдет в макет. (Так как ваш макет содержит HEAD и теги BODY, вы можете просто добавить соответствующий код в тег HEAD.) Вам просто нужно убедиться, что вы передаете информацию, необходимую для макета, с контроллера через ViewBag/View.Model/ViewData. Таким образом, ваш макет будет включать в себя следующее:

<head>
    <link rel="stylesheet" href="/css/@ViewBag.UserName/.css"/>
</head>

и ваш контроллер будет включать логику для заполнения ViewBag.UserName:

ViewBag.UserName = Session["UserName"];

(В идеале вы бы использовали сильно типизированную модель представления, и я бы порекомендовал вам воздерживаться от использования сеанса для чего-либо, поскольку его преимущества малы по сравнению с альтернативами и стоят за большие деньги для архитектуры... вместо этого я бы просто рекомендовал хранение некоторого зашифрованного файла cookie в браузере, который содержит имя пользователя или что-то еще, которое вы можете использовать при каждой загрузке страницы для извлечения пользовательского объекта из кеша/db/service.)