Muy, muy, muy molesto… ASP.NET MVC3 corriendo sobre un servidor web configurado en español (cultura es-ES).
Con la tabla de rutas estándar, cuatro acciones como las siguientes
- [HttpPost]
- public ActionResult Index(DoubleModel model)
- {
- ViewBag.Valor = model.Valor;
- return View("Resultado");
- }
- [HttpPost]
- public ActionResult IndexSoloDouble(double? valor)
- {
- ViewBag.Valor = valor;
- return View("Resultado");
- }
- public ActionResult IndexRuta(double? id)
- {
- ViewBag.Valor = id;
- return View("Resultado");
- }
- public ActionResult IndexGet(double? valor)
- {
- ViewBag.Valor = valor;
- return View("Resultado");
- }
La clase DoubleMode simplemente tiene una propiedad double llamada Valor.
- La primera acción recibe un double via POST dentro de un modelo
- La segunda recibe un double via POST sólo
- La tercera acción recibe un double como parámetro de ruta
- La cuarta acción recibe un double via querystring
Pues bien:
- La primera acción acepta 12,10 pero no 12.10 (usa la cultura del servidor web)
- La segunda acción acepta 12,10 pero no 12.10 (usa la cultura del servidor web)
- La tercera acción acepta 12.10 pero no 12,10 (usa la cultura ¿invariante?)
- La cuarta acción acepta 12.10 pero no 12, 10 (como la tercera, parece usar la invariante).
Aquí tenéis el código de la vista usada para enviar los datos:
- @using MvcApplication1.Models
- @model DoubleModel
- @{
- ViewBag.Title = "Index";
- }
- <h2>Index</h2>
- @{
- Html.EnableClientValidation(false);
- }
- @using (Html.BeginForm()) {
- <div>
- Introduce el valor:
- <input type="hidden" name="Valor" value="12,10" />
- </div>
- <input type="submit" value="enviar 12,10 via POST" />
- }
- @using (Html.BeginForm()) {
- <div>
- <input type="hidden" name="Valor" value="12.10" />
- </div>
- <input type="submit" value="enviar 12.10 via POST" />
- }
- @using (Html.BeginForm("IndexSoloDouble","Home"))
- {
- <div>
- <input type="hidden" name="Valor" value="12,10" />
- </div>
- <input type="submit" value="enviar 12,10 via POST (solo double)" />
- }
- @using (Html.BeginForm("IndexSoloDouble", "Home"))
- {
- <div>
- <input type="hidden" name="Valor" value="12.10" />
- </div>
- <input type="submit" value="enviar 12.10 via POST (solo double)" />
- }
- @Html.ActionLink("Enviar 12,10 via GET", "IndexGet", new { valor = "12,10" }); | @Html.ActionLink("Enviar 12.10 via GET", "IndexGet", new { valor = "12.10" });
- <br/>
- @Html.ActionLink("Enviar 12,10 en ruta", "IndexRuta", new { id = "12,10" }); | @Html.ActionLink("Enviar 12.10 en ruta", "IndexRuta", new { id = "12.10" });
El problema parece estar en los ValueProviders de URL (RouteDataValueProvider y QueryStringVaueProvider), que parecen ignorar la cultura, mientras que el ValueProvider de POST (FormValueProvider) la respeta.
Lo dicho… muy molesto 🙁
Saludos!
Editado 16:43. PD: Iván Loire (@ivanloire) me comenta que este comportamiento es por diseño: los locales NO se aplican en los parámetros de URL para mantener las URLs canónicas. Si usáramos los locales del usuario podríamos tener URLs distintas que sirviesen el mismo contenido, y URLs válidas para un usuario podrían no serlo para otro. Es por ello que en MVC han optado por usar la cultura invariante en los ValueProviders de la URL. Dejo dos enlaces que me ha pasado Iván a dos posts donde se explica esto: http://bit.ly/tIZX64 y http://bit.ly/vCgbmf