Compartir a través de


Knockout.js y Linq.js con ASP.NET MVC y Entity Framework (es-ES)

http://www.nicholls.co/blog/image.axd?picture=image_thumb_147.png

El día de hoy visualizarán como pueden mejorar la manera en que desarrollan sus proyectos (cuando solo usan Javascript y jQuery) empleando una base limpia que es Knockout.js que nos permite generar interfaces de usuario robustas sin perdernos en una telaraña de controladores de eventos y actualizaciones manuales del DOM (Esto no quiere decir que vamos a dejar de utilizar nuestro querido jQuery ya que estas librerías tienen propósitos totalmente distintos).

Diapositiva de la Charla

https://cdn3.iconfinder.com/data/icons/tango-icon-library/48/x-office-presentation-128.png

Ejemplos de código empleados en la Charla

https://cdn2.iconfinder.com/data/icons/snipicons/500/application-code-m-128.pnghttps://cdn2.iconfinder.com/data/icons/snipicons/500/application-code-m-128.png

Mediante Knockout nosotros obtenemos una serie de ventajas importantes tales como:

http://www.nicholls.co/blog/image.axd?picture=image_thumb_152.png

Como podemos observar en la documentación, Knockout emplea observadores para mantener sincronizada la interfaz de usuario con el modelo de datos que nosotros configuremos, junto con un conjunto de enlaces declarativos para permitir un desarrollo mucho más productivo los cuales vamos a entender más adelante en este artículo.

Configurando nuestro Proyecto ASP.NET MVC

Inicialmente** **vamos a dejar todo listo en nuestro proyecto referente al lado del Servidor y ver como ASP.NET MVC va a comunicarse (transferir los datos) con Knockout.

  • Configurando Knockout

Procedemos a instalar Knockout desde el Manage NuGet Packages.

http://www.nicholls.co/blog/image.axd?picture=image_thumb_154.png 

Y registramos nuestro archivo Javascript en nuestra Clase BundleConfig (ubicada dentro de la carpeta App_Start)

BundleConfig.cs

  1. // For more information on Bundling, visit http://go.microsoft.com/fwlink/?LinkId=254725
  2. publicstaticvoid RegisterBundles(BundleCollection bundles)
  3. {
  4.     //...Aquí configuramos nuestras técnicas para agrupar y minificar los archivos
  5.     //para reducir el numero de Peticiones al Servidor desde el Navegador y reducir el codigo de los archivos
  6.     //permitiendo mejorar el tiempo de carga de nuestras páginas Web
  7.     //Esto no tiene que nada que ver con la seguridad porque el javascript minificado se puede visualizar tal cual fue escrito http://jsbeautifier.org/
  8.  
  9.     bundles.Add(newScriptBundle("~/bundles/knockout").Include(
  10.                 "~/Scripts/knockout-{version}.js"));
  11.  
  12.     //...Otros archivos JS y CSS
  13. }

Como se explica en el siguiente artículo http://sebys.com.ar/2013/08/21/asp-net-mvc-y-knockout-viewmodel/, existen 2 opciones de pasar los datos a la Vista al momento de inicializar el modelo de Knockout, mediante peticiones AJAX (Generalmente un GET) y cuando se renderiza la Vista por el Controlador (Se carga la página Web en nuestro navegador) , la primera principalmente nos es útil cuando los datos cambian totalmente dependiendo de una acción del usuario (Por ejemplo cuando se selecciona un departamento y se deben de cargar las ciudades respectivas), de lo contrario la segunda opción siempre será la más optima porque los datos están siendo obtenidos en la misma petición de la Página Web, reduciendo por tanto el número de peticiones del navegador hacia nuestro Servidor (Influye directamente en el tiempo de carga de la página web) lo cual es un factor totalmente importante con respecto a la Experiencia de Usuario.

Para pasar los datos de nuestros Modelos a nuestras Vistas vamos a emplear el Método de Extensión propuesto en el artículo anterior, para esto procedemos a dar Click derecho en nuestro Proyecto y añadimos la Carpeta App_Code.

http://www.nicholls.co/blog/image.axd?picture=image_thumb_155.pngEsta Carpeta nos permite la creación de Clases personalizadas, en donde el código se compilará automáticamente en tiempo de ejecución. Por lo tanto nuestra Clase se verá de la siguiente manera:

JsonExtension.cs

  1. using System.IO;
  2. using Newtonsoft.Json;
  3.  
  4. namespace Knockout.Mvc.Extensions
  5. {
  6.     publicstaticclassJsonExtension
  7.     {
  8.         publicstaticstring ToJson(thisobject obj)
  9.         {
  10.             JsonSerializer js = JsonSerializer.Create(newJsonSerializerSettings());
  11.             var jw = newStringWriter();
  12.             js.Serialize(jw, obj);
  13.             return jw.ToString();
  14.         }
  15.     }
  16. }

Este Método de Extensión nos permitirá convertir nuestras Entidades a un objeto JSON el cual será manipulado posteriormente por Knockout.

  • Configurando el Modelo para ser Serializado

Al Serializar una Entidad a un objeto JSON con el Método anterior debemos de tener cuidado de no incluir las relaciones que tenga esta con Colecciones de otras Entidades aplicando el atributo JsonIgnore, como por ejemplo:

UserList.cs

  1. using Newtonsoft.Json;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.ComponentModel.DataAnnotations;
  5. using System.ComponentModel.DataAnnotations.Schema;
  6. namespace WebApp.Models
  7. {
  8.     publicclassUserList
  9.     {
  10.         public UserList()
  11.         {
  12.             //La clase HashSet proporciona operaciones de conjuntos de alto rendimiento.
  13.             //Un conjunto es una coleccion que no contiene elementos duplicados, y cuyos elementos no tienen un orden en particular.
  14.             Products = newHashSet<Product>();
  15.             UserListFriends = newHashSet<UserListFriend>();
  16.         }
  17.  
  18.         [Key]
  19.         [DatabaseGenerated(DatabaseGeneratedOption.Identity)]
  20.         publicint IdUserList { get; set; }
  21.  
  22.         publicDateTime CreationDate { get; set; }
  23.  
  24.         publicint IdUser { get; set; }
  25.  
  26.         [ForeignKey("IdUser")]
  27.         publicUser User { get; set; }
  28.  
  29.         [JsonIgnore]
  30.         publicvirtualICollection<Product> Products { get; set; }
  31.  
  32.         [JsonIgnore]
  33.         publicvirtualICollection<UserListFriend> UserListFriends { get; set; }
  34.     }
  35. }

Para entender mejor acerca de la serialización de objetos en JSON podemos observar el siguiente artículo http://julitogtu.com/2013/11/12/asp-net-web-api-personalizando-la-serializacion-de-los-objetos-en-json/

Un dato curioso con Entity Framework para los que siempre nos encargamos de realizar la Normalización  de nuestra Base de Datos, es que si por ejemplo tenemos 2 Tablas las cuales tienen una relación de muchos a muchos es obvio que tenemos que crear un Tabla intermedia la cual nos sirva de puente entre las dos, registrando la Clave Primaria de ambas Tablas. Si en nuestro caso no vamos a incluir un campo adicional a esta Tabla intermedia podemos dejarle la labor al Entity Framework de  crearla automáticamente. Para que tengamos un ejemplo claro la Tabla anterior UserLists tiene una relación de muchos a muchos con la Tabla Products (una Lista de Usuario puede tener muchos Productos asignados y un mismo Producto puede estar en muchas Listas de Usuario), por lo cual en nuestro modelo un Producto sería por ejemplo:

Product.cs

  1. using Newtonsoft.Json;
  2. using System.Collections.Generic;
  3. using System.ComponentModel.DataAnnotations;
  4. using System.ComponentModel.DataAnnotations.Schema;
  5. namespace WebApp.Models
  6. {
  7.     publicclassProduct
  8.     {
  9.         public Product()
  10.         {
  11.             UserLists = newHashSet<UserList>();
  12.         }
  13.         [Key]
  14.         [DatabaseGenerated(DatabaseGeneratedOption.Identity)]
  15.         publicint IdProduct { get; set; }
  16.  
  17.         publicstring Name { get; set; }
  18.  
  19.         publicstring Image { get; set; }
  20.  
  21.         [JsonIgnore]
  22.         publicvirtualICollection<UserList> UserLists { get; set; }
  23.     }
  24. }

Y ya Entity Framework se encargaría en este caso de crear la Tabla Intermedia UserListProducts.

  • Detalle Importante para tener en Cuenta de Entity Framework en Nuestros Controladores de WebAPI

Supongamos que en una acción del Usuario vamos a registrar su Lista, la cual definimos en nuestro Modelo (UserList). Esto lo vamos a realizar mediante un Controlador de WebAPI al cual le vamos a enviar los datos que requiere una entidad UserList para crearse, en un POST con AJAX. El usuario ya ha seleccionado con Knockout su lista de productos y su lista de amigos, los productos fueron obtenidos de la Base de Datos con nuestro Método de Extensión (definido previamente) al momento de cargarse la página Web, en cambio los amigos fueron obtenidos mediante una petición AJAX a una URL externa como por ejemplo con Facebook.

Como la lista de amigos son una Colección de registros totalmente nuevos a la Tabla UserListFriends con Entity Framework no existe ningún problema, al momento de insertar la nueva Lista (UserList) y guardar los cambios ( context.SaveChanges(); ) se insertará también la lista de amigos en la Base de Datos, pero el caso de la Colección de Productos es totalmente distinto y ya les voy a explicar el porque.

Entity Framework al realizar un Select a nuestra Base de Datos se encarga  de asociar la respuesta con las Entidades definidas en nuestro Modelo y el Contexto por lo tanto le hace un seguimiento a estas Entidades. Como la lista de Productos que le mandaremos a Knockout inicialmente pertenecen a un Contexto diferente al que vamos a emplear en el Controlador de WebAPI, a pesar que de los Productos ya existen en la Base de Datos y estén con la misma Clave Primaria, Entity Framework los tomará como nuevas Entidades y si guardamos serán registros totalmente nuevos, para evitar esto necesitamos primero realizar un Select de los Productos antes de asociar estos con nuestra Lista de Usuario, para que el Contexto entienda que ya existen:  

UserListsController.cs

  1. namespace WebApp.Controllers
  2. {
  3.     publicclassUserListsController : ApiController
  4.     {
  5.         // POST api/userlists
  6.         publicHttpResponseMessage Post(UserList userlist)
  7.         {
  8.             if (ModelState.IsValid)
  9.             {
  10.                 var Products = userlist.Products;
  11.                 userlist.Products = newList<Product>();
  12.                 using (DataContext context = newDataContext())
  13.                 {
  14.                     foreach (var product in Products)
  15.                     {
  16.                         userlist.Products.Add(context.Products.Single(x => x.IdProduct == product.IdProduct));
  17.                     }
  18.                     userlist.User = context.Users.Single(x => x.IdUser == userlist.IdUser);
  19.                     userlist.CreationDate = DateTime.UtcNow.AddHours(-5);
  20.                     context.UserLists.Add(userlist);
  21.                     context.SaveChanges();
  22.                 }                
  23.                 return Request.CreateResponse(HttpStatusCode.Created, userlist);
  24.             }
  25.             else
  26.             {
  27.                 return Request.CreateErrorResponse(HttpStatusCode.BadRequest, ModelState);
  28.             }
  29.         }
  30.     }
  31. }

Se aceptan sugerencias de maneras más optimas para realizar este procedimiento y por lo cual actualizar este artículo, la otra opción sería utilizar el Attach de Entity Framework para un Objeto de una misma Entidad o en su defecto que se llamen Procedimientos Almacenados con el Entity Framework como podemos observar en el siguiente artículo http://jhonnyslopez.wordpress.com/2012/04/21/procedimientos-almacenados-transacciones-en-entity-framework-4-3-1-con-c-y-visual-studio-11/

 Iniciando con Knockout.js y Linq.js

http://mooretech.us/mvcko/d1.png

Fuente: http://it.toolbox.com/blogs/rymoore/aspnet-mvc-and-knockout-mvc-vs-mvvm-54423

Knockout al aplicar el Modelo de Presentación (ViewModel) emplea observadores para enlazar los datos de nuestro Modelo de manera bidireccional a nuestra Vista, esto en sí nos permite realizar un seguimiento de los cambios que se realicen en la Vista y tanto el Modelo de Presentación como la misma Vista se mantendrán actualizados sin que nosotros tengamos que realizar alguna manipulación del DOM manualmente. Por cierto, No todo en nuestro Modelo de Presentación de Knockout tiene que ser observable, solo aquello que se requiera.

Ahora veamos un ejemplo para entender todo esto:

http://www.nicholls.co/blog/image.axd?picture=image_thumb_162.png

Como pueden observar en este sencillo ejemplo ya estamos haciendo uso de nuestro Método de Extensión con una Lista de Productos que será pasada por el Controlador a nuestra Vista, por lo cual debemos importar tanto el Namespace de nuestro Modelo como el de la Clase de nuestro método.

Aunque el código HTML con jQuery es más corto que el de Knockout en este ejemplo, este último presenta una serie de ventajas:

  • La mayoría de veces no trabajamos solos y por buenas practicas nuestro Javascript quedará en un archivo externo (Especialmente para depurarlo en navegadores como Chrome en la fase de desarrollo) por lo cual si el diseñador decide cambiar el html lo puede hacer perfectamente sin tener que modificar nuestro javascript y mucho menos tener que buscar en todo el archivo la parte donde se inserta el código.

  • Como comenté anteriormente no todo en Knockout tiene que ser observable, solo aquello que nos importe tener actualizado en nuestro Modelo de Presentación y sincronizado con la Vista será observable. En este caso se esta obteniendo una Lista de Productos los cuales no van a ser alterados, pero si esta lista fuera a ser modificada más adelante y quisiéramos que nuestro html se mantuviera actualizado solo basta con hacer observable nuestro Array:

    self.Products = ko.observableArray(@Html.Raw((ViewBag.Products as List<Product>).ToJson()));
    
  • Y el punto más relevante es que logramos identificar en el html mediante el atributo data-bind (el cual es el que emplea Knockout para enlazar los datos a la Vista de manera declarativa), cual es el evento Click al que apunta en nuestro ViewModel,como el hecho de agregarle la Clase “selected” al Producto que hayamos escogido y asignarle el source a la imagen con el nombre de una propiedad de nuestro Producto definido anteriormente en el Modelo de ASP.NET MVC. Una de las cosas más interesantes es la posibilidad que nos da Javascript para saber si un elemento existe (diferente de null, undefined y false al mismo tiempo) con la doble negación (!!) además del manejo del operador ternario para retornar un valor dependiendo de la condición** **y Gracias a Knockout mediante el Patrón Observador,  que implementa con funciones como ko.observable(),nuestra Vista estará constantemente actualizada.

Ahora veamos un ejemplo de como sería con Knockout, jQuery y Linq.js implementar un buscador y seleccionador de amigos:

http://www.nicholls.co/blog/image.axd?picture=image_thumb_163.png

 

  1. @*El buscador esta asociado a una propiedad observable el cual se actualiza a medida que se escribe*@
  2. <input placeholder="Buscar" data-bind="value: searchFacebook, valueUpdate: 'afterkeydown'"/>
  3. @*El foreach retorna los amigos que coincidan con el buscador mediante la funcion ko.computed la cual se actualiza cuando una de sus dependencias observables cambia*@
  4. <div data-bind="foreach: SearchFacebookFriends">
  5.     @*Con jQuery sabemos si el amigo ha sido seleccionado (si se encuentra en el array observable de amigos elegidos)*@
  6.     <a href="javascript:void(0)" data-bind="css: { selected: $.inArray($data, $root.chosenFriends()) > -1 }, click: $root.choseFriend">
  7.         <img width="54" height="54" data-bind="attr:{ src: '//graph.facebook.com/'+ FacebookId +'/picture?type=square' }"/>
  8.         <span data-bind="text: Name"></span>
  9.     </a>
  10. </div>
  11. <script>
  12.     function WebAppViewModel() {
  13.         var self = this;
  14.         self.searchFacebook = ko.observable();
  15.         self.facebookFriends = ko.observableArray();
  16.         self.chosenFriends = ko.observableArray();
  17.         //Obtenemos el listado de amigos de una fuente como Facebook
  18.         FB.api('/me/friends?fields=id,name', function (response) {
  19.             if (response.data) {
  20.                 for (var i = 0; i < response.data.length; i++) {
  21.                     var friend = response.data[i];
  22.                     self.facebookFriends.push({ Name: friend.name, FacebookId: friend.id });
  23.                 }
  24.             }
  25.         });
  26.         //Cuando el buscador cambia mediante Linq.js retornamos los usuarios que coincidan con el termino de busqueda, si no hay nada se muestran todos y ordenados por el nombre
  27.         self.SearchFacebookFriends = ko.computed(function () {
  28.             var searchterm = self.searchFacebook();
  29.             if (searchterm) {
  30.                 return Enumerable.From(self.facebookFriends())
  31.                                  .Where(function (x) { return x.Name.toLowerCase().indexOf(searchterm.toLowerCase()) !== -1 })
  32.                                  .OrderBy(function (x) { return x.Name })
  33.                                  .ToArray();
  34.             }
  35.             else {
  36.                 return Enumerable.From(self.facebookFriends())
  37.                                  .OrderBy(function (x) { return x.Name })
  38.                                  .ToArray();
  39.             }
  40.         });
  41.         //Seleccionamos o Deseleccionamos a un amigo
  42.         self.choseFriend = function (friend, event) {
  43.             $target = $(event.target.parentElement);
  44.             if (!$target.hasClass("selected")) {
  45.                 self.chosenFriends.push(friend);
  46.             } else {
  47.                 self.chosenFriends.remove(friend);
  48.             }
  49.         };
  50.     }
  51.     ko.applyBindings(new WebAppViewModel());
  52. </script>

Como podemos observar Linq.js nos permite manipular arreglos muy fácilmente, muy parecido a como lo hacemos habitualmente con Linq y expresiones Lambda en C# pero esta vez desde nuestro Javascript.

Como ejemplo final veamos como sería realizar una petición AJAX a un Controlador de WebAPI definido previamente, enviando en un solo POST la Lista del Usuario (UserList) la cual incluye la lista de amigos (UserListFriends) y la lista de productos (Products) seleccionados con Knockout.

  1. self.PostData = function () {
  2.     $.post("/api/userlists/", {
  3.         IdUser: "Id del Usuario",
  4.         Products: self.chosenProducts(),
  5.         UserListFriends: self.chosenFriends()
  6.     }, function (userlist) {
  7.         //Respuesta Exitosa
  8.     }).fail(function () {
  9.         //Error
  10.     });
  11. };

Como podemos observar es tan fácil como indicar los campos definidos en la Entidad (UserList) de nuestro Modelo de ASP.NET MVC para que el Controlador pueda identificar los campos y asociarlos a las propiedades de nuestra Entidad.

Les recomiendo que realicen los tutoriales que tienen en el Sitio Web de Knockout http://learn.knockoutjs.com/.

Y como siempre que lean la documentación a medida que practican por ustedes mismos, ya que esta ha sido una breve introducción a esta poderosa herramienta http://knockoutjs.com/documentation/introduction.html

Esperamos que con este artículo ustedes hayan podido ver un poquito de lo que es el mundo de Knockout.js y les haya dado esa curiosidad de aprender y conocer más a fondo esta excelente librería para que la puedan poner en practica en sus proyectos, todo esto con el fin de que seamos mucho más productivos.