Nota: Este post ha sido importado de mi blog de geeks.ms. Es posible que algo no se vea del todo "correctamente". En cualquier caso puedes acceder a la versión original aquí
La verdad es que el tema de los enums y ASP.NET MVC da para hablar bastante (yo mismo hice un post hace ni mucho). Pero hace algunos días mi buen amigo y a veces rival, Marc Rubiño publicó en su blog un interesante artículo sobre como crear combos que mostrasen valores de enums.
En este post voy a mostrar una técnica parecida, pero a través de las data list, un concepto nuevo de HTML5 que como pasa muchas veces está recibiendo menos atención de la que merece. Y es que las data list nos dan una manera fácil y sencilla de tener cajas de texto que se autocompleten.
Qué son las data lists de HTML5?
Bueno, pues como ya debes deducir de su nombre las data list no es nada más que la posibilidad de definir un concepto nuevo, que es esto: una lista de datos.
Una lista de datos se usa como fuente de datos para un control que pueda tener una y lo bueno es que el viejo y a veces injustamente denostado entran en esta categoría. Honestamente no entiendo como el
La definición de una data list es muy simple:
<datalist id="sexo">
<option value="N">Nadaoption>
<option value="P">Pocooption>
<option value="M">Muchooption>
<option value="D">Demasiadooption>
datalist>
Ahora en HTML5 podemos usar el atributo list de la etiqueta para asociar a un (p. ej. una caja de texto) una lista de valores:
<input type="text" id="txtSexo" list="sexo" placeholder="Sexo" autocomplete="on" />
Básicamente tan solo se trata de usar el atributo list con el valor del id de la data list a usar.
El atributo id es requerido por Chrome (si no se aplica al Chrome ignora el atributo list). Este es el resultado en Chrome, Opera e IE10:
Podemos ver las diferencias entre navegadores:
- Chrome nos muestra el value a la izquierda y la descripción a la derecha
- Opera tan solo nos muestra el value
- IE10 tan solo nos muestra la descripción
Sin duda el mejor mecanismo, en mi opinión, es el de Chrome, luego el de Opera y por último el de IE10. No he podido probar FF ni Safari por no tenerlos a mano en el momento de escribir el post.
Igual te sorprende que prefiera que el desplegable muestre N, P, M y D como hace Opera en lugar de Nada, Poco,… que muestra IE10. Eso es porque el valor que se debe teclear en el textbox es el de value. El valor de value es el que se almacena en el textbox, y por lo tanto es el que debe teclearse (a diferencia de la
Ten presente siempre que estamos usando un por lo que el usuario puede entrar lo que quiera. No es como una
Un ejemplo.
Veamos como podríamos usar esto con enums y MVC…
Lo primero es definirnos un enum:
public enum Beers
{
Estrella_Damm,
Voll_Damm,
Moritz,
Epidor,
Mahou,
Cruzcampo
}
Ahora vamos a extender la clase HtmlHelper con un método de extensión nuevo que
llamaremos TextBoxEnumerableFor:
public static class HtmlHelperEnumExtensions
{
public static IHtmlString TextBoxEnumerableFor<TModel, TProperty>(this HtmlHelper<TModel> helper, Type enumerationType, Expression<Func<TModel, TProperty>> expression)
{
var httpContext = helper.ViewContext.HttpContext;
var html = new StringBuilder();
if (httpContext.Items[enumerationType.FullName] == null)
{
httpContext.Items.Add(enumerationType.FullName, GenerateDataList(enumerationType, html));
}
var textbox = helper.TextBoxFor(expression, new {list = enumerationType.FullName});
html.AppendLine(textbox.ToString());
return MvcHtmlString.Create(html.ToString());
}
private static string GenerateDataList(Type enumerationType, StringBuilder html)
{
var id = enumerationType.FullName;
html.AppendFormat(@"
html.AppendLine();
foreach (var item in Enum.GetNames(enumerationType))
{
html.AppendFormat(@"{0}, item.Replace(‘_’, ‘ ‘));
html.AppendLine();
}
html.AppendLine("");
<p style="margin: 0px">
 
</p>
<p style="margin: 0px">
            <span style="color: #569cd6">return</span> <span style="color: white">id</span>;
</p>
<p style="margin: 0px">
        }
</p>
<p style="margin: 0px">
    }
</p></p></div>
<p>
Aspectos a comentar de este código:
</p>
<ol>
<li>
Uso HttpContext.Items para “hacer un seguimiento” de si ya se ha creado una datalist. Esto es para no definir en el mismo HTML dos veces la misma datalist si se generan dos (o más) cajas de texto vinculadas al mismo enum.
</li>
<li>
Sustituyo los guiones bajos (_) por un espacio. Eso debería tenerse en cuenta luego al recibir los datos.
</li>
</ol>
<p>
Este código es muy sencillo y realmente debería sobrecargarse el método para dar más opciones (p. ej. especificar atributos html adicionales que ahora no es posible). Pero como ejemplo, creo que sirve.
</p>
<p>
Su uso es muy sencillo:
</p>
<div style="font-size: 10pt; font-family: consolas; background: #1e1e1e; color: #dcdcdc">
<p style="margin: 0px">
<span style="background: #ffffb3; color: black">@</span><span style="color: #569cd6">using</span> <span style="color: white">MvcApplication1</span>
</p>
<p style="margin: 0px">
<span style="background: #ffffb3; color: black">@</span><span style="color: #569cd6">using</span> <span style="color: white">MvcApplication1</span><span style="color: #b4b4b4">.</span><span style="color: white">Models</span>
</p>
<p style="margin: 0px">
<span style="background: #ffffb3; color: black">@model </span><span style="color: white">MvcApplication1</span><span style="color: #b4b4b4">.</span><span style="color: white">Models</span><span style="color: #b4b4b4">.</span><span style="color: #4ec9b0">BeerModel</span>
</p>
<p style="margin: 0px">
 
</p>
<p style="margin: 0px">
<span style="background: #ffffb3; color: black">@{</span>
</p>
<p style="margin: 0px">
    <span style="color: white">ViewBag</span><span style="color: #b4b4b4">.</span><span style="color: white">Title</span> <span style="color: #b4b4b4">=</span> <span style="color: #d69d85">"Index"</span>;
</p>
<p style="margin: 0px">
<span style="background: #ffffb3; color: black">}</span>
</p>
<p style="margin: 0px">
Entra el nombre de tu cerveza preferida aquí:
</p>
<p style="margin: 0px">
 
</p>
<p style="margin: 0px">
<span style="background: #ffffb3; color: black">@</span><span style="color: white">Html</span><span style="color: #b4b4b4">.</span><span style="color: white">TextBoxEnumerableFor</span>(<span style="color: #569cd6">typeof</span>(<span style="color: #b8d7a3">Beers</span>), <span style="color: white">m</span><span style="color: #b4b4b4">=></span><span style="color: white">m</span><span style="color: #b4b4b4">.</span><span style="color: white">BeerName</span>)
</p></p>
</div>
<p>
BeerName es una propiedad (string) del modelo BeerModel. Y este es el HTML generado:
</p>
<pre class="csharpcode">Entra el nombre de tu cerveza preferida aquí:
<datalist id=“MvcApplication1.Models.Beers”>
<option>Estrella Damm</option>
<option>Voll Damm</option>
<option>Moritz</option>
<option>Epidor</option>
<option>Mahou</option>
<option>Cruzcampo</option>
</datalist>
<input id=“BeerName”
list=“MvcApplication1.Models.Beers”
name=“BeerName”
type=“text” value="" />
<p>
No generamos atributo value, en este caso dicho atributo toma el valor del propio texto del option.
</p>
<p>
Y un pantallazo de como se ve en Opera:
</p>
<p>
<a href="http://geeks.ms/cfs-file.ashx/__key/CommunityServer.Blogs.Components.WeblogFiles/etomas/image_5F00_54A2AD16.png"><img title="image" style="border-top: 0px; border-right: 0px; border-bottom: 0px; border-left: 0px; display: inline" border="0" alt="image" src="http://geeks.ms/cfs-file.ashx/__key/CommunityServer.Blogs.Components.WeblogFiles/etomas/image_5F00_thumb_5F00_5285AE4D.png" width="244" height="107" /></a>
</p>
<p>
Puedes ver como se usa los elementos del enum para mostrar una sugerencia de autocompletado del textbox. Pero recuerda: el usuario <strong>puede entrar el valor que quiera</strong>.
</p></p>
<p>
Como digo, el código se puede mejorar mucho, pero como idea y ejemplo creo que es suficiente.
</p>
<p>
Un saludo!
</p>