TempData Wrapper
Поскольку я использую одни и те же ключи для TempData снова и снова, я хотел бы упростить отслеживание этих ключей. Было бы здорово, если бы в конце концов я мог написать что-то вроде этого:
MyTempData.Message = "my message";
вместо
TempData["Message"] = "my message";
4 ответа
Похоже, вы хотите строго типизированные TempData. Один из способов сделать это - написать метод расширения для контроллера:
public static class ControllerExtensions
{
public string GetMessage(this Controller instance)
{
string result = instance.TempData["Message"] as string;
if (result == null)
{
result = "some default value or throw null argument exception";
}
return result;
}
public void SetMessage(this Controller instance, string value)
{
instance.TempData["Message"] = value;
}
}
Довольно низкотехнологичный вариант, но если вы просто хотите отслеживать используемые ключи, вы можете просто создать несколько констант:
public static class TempDataKeys
{
public const string Message = "Message";
public const string Warning = "Warning";
public const string Error = "Error";
// etc
}
затем:
TempData[TempDataKeys.Message] = "Some message";
Вы могли бы подражать что ViewBag
делает.
Во-первых, он использует внутренний запечатанный класс DynamicViewDataDictionary
, По сути, я просто собираюсь сделать версию этого для TempData. Затем мы можем использовать методы расширения, чтобы сделать его доступным в Controller
, WebViewPage
, так далее.
public sealed class DynamicTempDataDictionary : DynamicObject
{
private readonly Func<TempDataDictionary> _tempDataThunk;
public DynamicTempDataDictionary(Func<TempDataDictionary> viewDataThunk)
{
_tempDataThunk = viewDataThunk;
}
private TempDataDictionary ViewData
{
get { return _tempDataThunk(); }
}
public override IEnumerable<string> GetDynamicMemberNames()
{
return ViewData.Keys;
}
public override bool TryGetMember(GetMemberBinder binder, out object result)
{
result = ViewData[binder.Name];
return true;
}
public override bool TrySetMember(SetMemberBinder binder, object value)
{
ViewData[binder.Name] = value;
return true;
}
}
public static class TempBagExtensions
{
private const string TempBagKey = "__CurrentTempBagDictionary";
public static dynamic TempBag(this ControllerBase controller)
{
return GetCurrentDictionary(controller.ControllerContext);
}
public static dynamic TempBag(this WebViewPage viewPage)
{
return GetCurrentDictionary(viewPage.ViewContext.Controller.ControllerContext);
}
private static DynamicTempDataDictionary GetCurrentDictionary(ControllerContext context)
{
var dictionary = context.HttpContext.Items[TempBagKey] as DynamicTempDataDictionary;
if (dictionary == null)
{
dictionary = new DynamicTempDataDictionary(() => context.Controller.TempData);
context.HttpContext.Items[TempBagKey] = dictionary;
}
return dictionary;
}
}
В вашем контроллере:
this.TempBag().SomeValue = "Test";
На ваш взгляд бритвы:
@this.TempBag().SomeValue
Если вы не думаете, что метод расширения достаточно чист, вы можете создать свой собственный базовый класс Controller, а также свои собственные базовые классы для бритвенных представлений, которые используют приятные, лаконичные свойства. ViewBag
,
Трудно сказать, на каком уровне вы хотите, чтобы временные данные сохранялись. Моя интерпретация заключается в том, что, похоже, вы хотели бы иметь общий набор свойств, доступный для всех ваших моделей представления.
Для этого унаследуйте от общего класса ViewModel.
public class BaseViewModel
{
public string Message{get;set;}
}
public class MyModel : BaseViewModel
{
public string MyUniquePropety{get;set;}
}
Если окажется, что вы хотите, чтобы данные сохранялись в сеансе, то внедрите синглтон и не забудьте добавить в вид представление @using для ссылки на него.