Использование тире (-) в параметрах ASP.MVC
<% using (Html.BeginForm("SubmitUserName")) { %>
<input type='text' name='user-name' />
<input type='submit' value='Send' />
<% } %>
Что должно быть сигнатурой соответствующего метода Action для принятия параметра user-name
?
public ActionResult SubmitUserName(string user-name) {...}
Подпись метода выше не работает по какой-либо причине; -)
Я знаю, что существует ActionNameAttribute
для обработки ситуации с тире в имени действия. Есть что-то вроде ParameterNameAttribute
?
Ответы
Ответ 1
Как все отметили, самым простым решением было бы не использовать тире. Если вам действительно нужна тире, вы можете создать свой собственный ActionFilterAttribute, чтобы обработать его.
Что-то вроде:
[AttributeUsage(AttributeTargets.Method, AllowMultiple = true)]
public class ParameterNameAttribute : ActionFilterAttribute
{
public string ViewParameterName { get; set; }
public string ActionParameterName { get; set; }
public override void OnActionExecuting(ActionExecutingContext filterContext)
{
if(filterContext.ActionParameters.ContainsKey(ViewParameterName))
{
var parameterValue = filterContext.ActionParameters[ViewParameterName];
filterContext.ActionParameters.Add(ActionParameterName, parameterValue);
}
}
}
Затем вы примените фильтр к соответствующему методу Action:
[ParameterName( ViewParameterName = "user-data", ActionParameterName = "userData")]
[ParameterName( ViewParameterName = "my-data", ActionParameterName = "myData" )]
public ActionResult About(string userData, string myData)
{
return View();
}
Вероятно, вы захотите улучшить атрибут ParameterNameAttribute для обработки верхнего/нижнего регистра, но это будет основная идея.
Ответ 2
Создайте псевдопараметр в первой строке метода действия:
public ActionResult SubmitUserName()
{
string userName = Request.Params["user-name"];
...
}
Ответ 3
Я нашел этот ответ полезным, но я точно не знаю, как помогает приведенный пример. Похоже, что просто "переименует" значение, которое все готово к приложению.
В моем случае я был отправлен внешней службой, которая могла бы отправить что-то вроде "body-plain", и я не мог управлять именем. Поэтому я изменил этот пример так:
[AttributeUsage(AttributeTargets.Method, AllowMultiple = true)]
public class ParameterNameMapAttribute : ActionFilterAttribute
{
public string InboundParameterName { get; set; }
public string ActionParameterName { get; set; }
public override void OnActionExecuting(ActionExecutingContext filterContext)
{
object value = filterContext.RequestContext.HttpContext.Request[InboundParameterName];
if (filterContext.ActionParameters.ContainsKey(ActionParameterName))
{
filterContext.ActionParameters[ActionParameterName] = value;
}
else
{
throw new Exception("Parameter not found on controller: " + ActionParameterName);
}
}
}
Это фактически принимает параметр "body-plain", например, и сопоставляет его с параметром ActionParameter I, определенным на моем контроллере. Например:
[ParameterNameMap(InboundParameterName = "body-plain", ActionParameterName = "bodyPlainText")]
[ParameterNameMap(InboundParameterName = "Thread-Topic", ActionParameterName = "alternateSubject")]
public HttpStatusCodeResult Process(string token, string timestamp, string signature, string subject, string sender, string recipient, string bodyPlainText, string alternateSubject)
{
Ответ 4
Я бы предложил сделать что-то вроде этого - если только не требуется использовать атрибут user-name
(или вы не привязываетесь к модели)
<% using (Html.BeginForm("SubmitUserName")) { %>
<%: Html.TextBoxFor(m => m.UserName) %>
<input type='submit' value='Send' />
<% } %>
Ответ 5
Не отвечая на фактический вопрос, основанный на рассматриваемой технологии, но в любом случае, мир движется вперед в некоторых областях; в AspNetCore.Mvc вы можете просто сделать:
[HttpGet()]
public ActionResult SubmitUserName( [FromHeader(Name = "user-Name")] string userName) {...}