A .NET Developer Primer for Single-Page Applications theo http://msdn.microsoft.com/

A .NET Developer Primer for Single-Page Applications

A majority of Microsoft .NET Framework developers have spent most of their professional lives on the server side, coding with C# or Visual Basic .NET when building Web applications. Of course, JavaScript has been used for simple things such as modal windows, validation, AJAX calls and so on. However, JavaScript (client-side code for the most part) has been leveraged as a utility language, and applications were largely driven from the server side.
Lately there’s been a huge trend of Web application code migrating from the server side to the client side (browser) to meet users’ expectations for fluid and responsive UX. With this being the case, a lot of .NET developers (especially in the enterprise) are dealing with an extreme amount of anxiety about JavaScript best practices, architecture, unit testing, maintainability and the recent explosion of different kinds of JavaScript libraries. Part of the trend of moving to the client side is the increasing use of single-page applications (SPAs). To say that SPA development is the future is an extreme understatement. SPAs are how some of the best applications on the Web offer fluid UX and responsiveness, while minimizing payloads (traffic) and round-trips to the server.
In this article, I’ll address the anxieties you might experience when making the transition from the server side into the SPA realm. The best way to deal with these anxieties is to embrace JavaScript as a first-class language just like any .NET language, such as C#, Visual Basic .NET, Python and so on.
Following are some fundamental principles of .NET development that are sometimes ignored or forgotten when developing apps in JavaScript:
  • Your code base is manageable in .NET because you’re decisive with class boundaries and where classes actually live within your projects.
  • You separate concerns, so you don’t have classes that are responsible for hundreds of different things with overlapping responsibilities.
  • You have reusable repositories, queries, entities (models) and data sources.
  • You put some thought into naming your classes and files so they’re more meaningful.
  • You practice good use of design patterns, coding conventions and organization.
Because this article is for .NET developers who are being introduced to the SPA world, I’ll incorporate the least number of frameworks possible to build a manageable SPA with sound architecture.

Creating an SPA in Seven Key Steps

Following are seven key steps to convert a new ASP.NET Web Application that was created with the out-of-the-box Visual Studio 2013 ASP.NET MVC template into an SPA (with references to the appropriate project files that can be found in the accompanying code download).
  1. Download and install the NuGet packages RequireJS, RequireJS text plug-in and Kendo UI Web.
  2. Add a configuration module (Northwind.Web/Scripts/app/main.js).
  3. Add an app module (Northwind.Web/Scripts/app/app.js).
  4. Add a router module (Northwind.Web/Scripts/app/router.js).
  5. Add an action and view both named Spa (Northwind.Web/Controllers/HomeController.cs and Northwind.Web/Views/Home/Spa.cshtml).
  6. Modify the _ViewStart.cshtml file so MVC will load views without using the _Layout.cshtml file by default (Northwind.Web/Views/_ViewStart.cshtml).
  7. Update the layout navigation (menu) links to match the new SPA-friendly URLs (Northwind.Web/Views/Shared/_Layout.cshtml).
After these seven steps have been carried out, your Web application project structure should look something like Figure 1.
ASP.NET MVC Project Structure
Figure 1 ASP.NET MVC Project Structure
I’ll show how to build an awesome SPA in ASP.NET MVC with the following JavaScript libraries, available via NuGet:
  • RequireJS (requirejs.org): This is a Java­Script file and module loader. RequireJS will provide #include/import/require APIs and the ability to load nested dependencies with dependency injection (DI). The RequireJS design approach uses the Asynchronous Module Definition (AMD) API for JavaScript modules, which helps to encapsulate pieces of code into useful units. It also provides an intuitive way to refer to other units of code (modules). RequireJS modules also follow the module pattern (bit.ly/18byc2Q). A simplified implementation of this pattern uses JavaScript functions for encapsulation. You’ll see this pattern in action later as all JavaScript modules will be wrapped within a “define” or “require” function.
  • Those familiar with DI and Inversion of Control (IoC) concepts can think of this as a client-side DI framework. If that’s as clear as mud at the moment, no worries—I’ll soon get into some coded illustrations where all this will make sense.
  • Text plug-in for RequireJS (bit.ly/1cd8lTZ): This will be used to remotely load chunks of HTML (views) into the SPA.
  • Entity Framework (bit.ly/1bKiZ9I): This is pretty self-explanatory, and because the focus of this article is on SPA, I won’t get too much into Entity Framework. However, if you’re new to this, there’s plenty of documentation available.
  • Kendo UI Web (bit.ly/t4VkVp): This is a comprehensive JavaScript/­HTML5 framework that encompasses Web UI Widgets, DataSources, templates, the Model-View-ViewModel (MVVM) pattern, SPAs, styling, and so on to help deliver a responsive and adaptive application that will look great.

Setting up the SPA Infrastructure

To show how to set up the SPA infrastructure, first I’ll explain how to create the RequireJS (config) module (Northwind.Web/Scripts/app/main.js). This module will be the app start-up entry point. If you’ve created a console app, you can think of this as the Main entry point in Program.cs. It basically contains the first class and the method that’s called when the SPA starts up. The main.js file basically serves as the SPA’s manifest and is where you’ll define where all things in the SPA are and their dependencies, if any. The code for RequireJS configuration is shown in Figure 2.
Figure 2 RequireJS Configuration
  1. require.config({
  2.   paths: {
  3.     // Packages
  4.     'jquery''/scripts/jquery-2.0.3.min',
  5.     'kendo''/scripts/kendo/2013.3.1119/kendo.web.min',
  6.     'text''/scripts/text',
  7.     'router''/scripts/app/router'
  8.   },
  9.   shim : {
  10.     'kendo' : ['jquery']
  11.   },
  12.   priority: ['text''router''app'],
  13.   jquery: '2.0.3',
  14.   waitSeconds: 30
  15. });
  16. require([
  17.   'app'
  18. ], function (app) {
  19.   app.initialize();
  20. });
In Figure 2, the paths property contains a list of where all the modules are located and their names. Shim is the name of a module defined previously. The shim property includes any dependencies the module may have. In this case, you’re loading a module named kendo and it has a dependency on a module named jquery, so if a module requires the kendo module, go ahead and load jQuery first, because jQuery has been defined as a dependency for the kendo module.
In Figure 2, the code “require([], function(){})” will load in the next module, which is the module I named app. Note that I’ve deliberately given meaningful names to modules.
So, how does your SPA know to invoke this module first? You configure this on the first landing page in the SPA with the data-main attribute in the script reference tag for RequireJS. I’ve specified that it run the main module (main.js). RequireJS will handle all the heavy lifting involved in loading this module; you just have to tell it which module to load first.
You have two options for SPA views that will be loaded into the SPA: standard HTML (*.html) or ASP.NET MVC Razor (*.cshtml) pages. Because this article is intended for .NET developers—and a lot of enterprises have server-side libraries and frameworks they’d like to continue using in their views—I’ll go with the latter option of creating Razor views.
I’ll start off by adding a view and name it Spa.cshtml, as mentioned previously. This view will basically load up the shell or all the HTML for the layout of the SPA. From this view, I’ll load in the other views (for example, About.cshtml, Contact.cshtml, Index.cshtml and so on) as the user navigates through the SPA, by swapping the views that replace all the HTML in the “content” div.
Creating the SPA Landing Page (Layout) (Northwind.Web/Views/Spa.cshtml) Because the Spa.cshtml view is the SPA’s landing page where you’ll load in all your other views, there won’t be much markup here, other than referencing the required style sheets and RequireJS. Note the data-main attribute in the following code, which tells RequireJS which module to load first:
  1. @{
  2.   ViewBag.Title = "Spa";
  3.   Layout = "~/Views/Shared/_Layout.cshtml";
  4. }
  5. <link href=
  6.   "~/Content/kendo/2013.3.1119/kendo.common.min.css" 
  7.   rel="stylesheet" />
  8. <link href=
  9.   "~/Content/kendo/2013.3.1119/kendo.bootstrap.min.css" 
  10.   rel="stylesheet" />
  11. <script src=
  12.   "@Url.Content("~/scripts/require.js")"
  13.   data-main="/scripts/app/main"></script>
  14. <div id="app"></div>
Adding an Action for the SPA Layout (Northwind.Web/­Controllers/HomeController.cs) To create and load the Spa.cshtml view, add an action and view:
  1. public ActionResult Spa()
  2. {
  3.   return View();
  4. }
Create the Application Module (Northwind.Web/Scripts/app/app.js) Here’s the Application module, responsible for initializing and starting the Kendo UI Router:
  1. define([
  2.     'router'
  3.   ], function (router) {
  4.     var initialize = function() {
  5.       router.start();
  6.     };
  7.     return {
  8.       initialize: initialize
  9.     };
  10.   });
Create the Router Module (Northwind.Web/Scripts/app/router.js) This is called by app.js. If you’re already familiar with ASP.NET MVC routes, it’s the same notion here. These are the SPA routes for your views. I’ll define all the routes for all the SPA views so when the user navigates through the SPA, the Kendo UI router will know what views to load into the SPA. See Listing 1 in the accompanying download.
The Kendo UI Router class is responsible for tracking the application state and navigating between the application states. The router integrates into the browser history using the fragment part of the URL (#page), making the application states bookmarkable and linkable. When a routable URL is clicked, the router kicks in and tells the application to put itself back into the state that was encoded into the route. The route definition is a string representing a path used to identify the state of the application the user wants to see. When a route definition is matched from the browser’s URL hash fragment, the route handler is called (see Figure 3).
Figure 3 Registered Route Definitions and Corresponding URLs
Registered Route (Definition)Actual Full (Bookmarkable) URL
/localhost:25061/home/spa/home/index
/home/indexlocalhost:25061/home/spa/#/home/index/home/about
/home/aboutlocalhost:25061/home/spa/#/home/about/home/contact
/home/contactlocalhost:25061/home/spa/#/home/contact/customer/index
/customer/indexlocalhost:25061/home/spa/#/customer/index
As for the Kendo UI layout widget, its name speaks for itself. You’re probably familiar with the ASP.NET Web Forms MasterPage or MVC layout included in the project when you create a new ASP.NET MVC Web Application. In this SPA project, it’s located at the path Northwind.Web/Views/Shared/_Layout.cshtml. There’s little difference between the Kendo UI layout and MVC layout, except the Kendo UI layout runs on the client side. Just as the layout worked on the server side, where the MVC runtime would swap out the content of the layout with other views, the Kendo UI layout works the same exact way. You swap out the view (content) of the Kendo UI layout using the showIn method. View contents (HTML) will be placed in the div with the ID “content,” which was passed into the Kendo UI layout when it was initialized. After initializing the layout, you then render it inside the div with the ID “app,” which is a div in the landing page (Northwind.Web/Views/Home/Spa.cshtml). I’ll review that shortly.
The loadView helper method takes in a view model, a view and—if needed—a callback to invoke once the view and view model binding takes place. Within the loadView method, you leverage the Kendo UI FX library to aesthetically enhance the UX by adding some simple animation to the view swapping process. This is done by sliding the current loaded view to the left, remotely loading in the new view and then sliding the new loaded view back to the center. Obviously, you can easily change this to a variety of different animations using the Kendo UI FX library. One of the key benefits of using the Kendo UI layout is shown when you invoke the showIn method to swap out views. It will ensure the view is unloaded, destroyed properly and removed from the browser’s DOM, thus ensuring the SPA can scale and is performant.
Edit the _ViewStart.cshtml View (Northwind.Web/Views/­_ViewStart.cshtml) Here’s how to configure all views to not use the ASP.NET MVC layout by default:
  1. @{
  2.   Layout = null;
  3. }
At this point, the SPA should be working. When clicking on any of the navigation links on the menu, you see the current content is being swapped out via AJAX thanks to the Kendo UI router and RequireJS.
These seven steps needed to convert a fresh ASP.NET Web Application into an SPA aren’t too bad, are they?
Now that the SPA is up and running, I’ll go ahead and do what most developers will end up doing with an SPA, which is adding some create, read, update and delete (CRUD) functionality.

Adding CRUD Functionality to the SPA

Here are the key steps needed to add a Customer grid view to the SPA (and the related project code files):
  • Add a CustomerController MVC controller (Northwind.Web/Controllers/CustomerController.cs).
  • Add a REST OData Customer Web API controller (Northwind.Web/Api/CustomerController.cs).
  • Add a Customer grid view (Northwind.Web/Views/­Customer/Index.cshtml).
  • Add a CustomerModel module (Northwind.Web/Scripts/app/models/CustomerModel).
  • Add a customerDatasource module for the Customer grid (Northwind.Web/Scripts/app/datasources/customer­Datasource.js).
  • Add an indexViewModel module for the Customer grid view (Northwind.Web/Scripts/app/viewModels/­indexViewModel.js).
Setting Up the Solution Structure with Entity Framework Figure 4 shows the solution structure, high­lighting three projects: Northwind.Data (1), Northwind.Entity (2) and Northwind.Web (3). I’ll briefly discuss each, along with Entity Framework Power Tools.
  • Northwind.Data: This includes everything related to the Entity Framework Object-Relational Mapping  (ORM) tool, for persistence.
  • Northwind.Entity: This includes domain entities, composed of Plain Old CLR Object (POCO) classes. These are all the persistent-ignorant domain objects.
  • Northwind.Web: This includes the ASP.NET MVC 5 Web Application, the presentation layer, where you’ll build out the SPA with two previously mentioned libraries—Kendo UI and RequireJS—and the rest of the server-side stack: Entity Framework, Web API and OData.
  • Entity Framework Power Tools: To create all the POCO entities and mappings (database-first), I used the Entity Framework Power Tools from the Entity Framework team (bit.ly/1cdobhk). After the code generation, all I did here is simply copy the entities into a separate project (Northwind.Entity) to address separation concerns.
A Best-Practice Solution Structure
Figure 4 A Best-Practice Solution Structure
Note: Both the Northwind SQL install script and a backup of the database are included in the downloadable source code under the Northwind.Web/App_Data folder (bit.ly/1cph5qc).
Now that the solution is set up to access the database, I’ll go ahead and write the MVC CustomerController.cs class to serve up the index and edit views. Because the controller’s only responsibility is to serve up an HTML view for the SPA, the code here will be minimal.
Creating MVC Customer Controller (Northwind.Web/­Controllers/CustomerController.cs) Here’s how to create the Customer controller with the actions for the index and edit views:
  1. public class CustomerController : Controller
  2. {
  3.   public ActionResult Index()
  4.   {
  5.     return View();
  6.   }
  7.   public ActionResult Edit()
  8.   {
  9.     return View();
  10.   }
  11. }
Creating the View with the Customers Grid (Northwind.Web/­Views/Customers/Index.cshtml)Figure 5 shows how to create the view with the Customers grid.
If the markup in Figure 5 isn’t familiar, don’t panic—it’s just the Kendo UI MVVM (HTML) markup. It simply configures an HTML element, in this case the div with an ID of “grid.” Later on when you bind this view to a view model with the Kendo UI MVVM framework, this markup will be converted to Kendo UI widgets. You can read more on this at bit.ly/1d2Bgfj.
Figure 5 Customer Grid View Markup with an MVVM Widget and Event Bindings
  1. <div class="demo-section">
  2.   <div class="k-content" style="width: 100%">
  3.     <div id="grid"
  4.       data-role="grid"
  5.       data-sortable="true"
  6.       data-pageable="true"
  7.       data-filterable="true"
  8.       data-editable="inline"
  9.       data-selectable="true"
  10.       data-toolbar='[ { template: kendo.template($("#toolbar").html()) } ]'
  11.       data-columns
  12. ='[
  13.         { field: "CustomerID", title: "ID", width: "75px" },
  14.         { field: "CompanyName", title: "Company"},
  15.         { field: "ContactName", title: "Contact" },
  16.         { field: "ContactTitle", title: "Title" },
  17.         { field: "Address" },
  18.         { field: "City" },
  19.         { field: "PostalCode" },
  20.         { field: "Country" },
  21.         { field: "Phone" },
  22.         { field: "Fax" } ]'
  23.       data-bind
  24. ="source: dataSource, events:
  25.         { change: onChange, dataBound: onDataBound }"
  26. >
  27.     </div>
  28.     <style scoped>
  29.     #grid .k-toolbar {
  30.       padding: 15px;
  31.     }
  32.     .toolbar {
  33.       float: right;
  34.     }
  35.     </style>
  36.   </div>
  37. </div>
  38. <script type="text/x-kendo-template" id="toolbar">
  39.   <div>
  40.     <div class="toolbar">
  41.       <span data-role="button" data-bind="click: edit">
  42.         <span class="k-icon k-i-tick"></span>Edit</span>
  43.       <span data-role="button" data-bind="click: destroy">
  44.         <span class="k-icon k-i-tick"></span>Delete</span>
  45.       <span data-role="button" data-bind="click: details">
  46.         <span class="k-icon k-i-tick"></span>Edit Details</span>
  47.     </div>
  48.     <div class="toolbar" style="display:none">
  49.       <span data-role="button" data-bind="click: save">
  50.         <span class="k-icon k-i-tick"></span>Save</span>
  51.       <span data-role="button" data-bind="click: cancel">
  52.         <span class="k-icon k-i-tick"></span>Cancel</span>
  53.     </div>
  54.   </div>
  55. </script>
Creating MVC (OData) Web API Customer Controller (Northwind.Web/Api/CustomerController.cs)Now I’ll show how to create the MVC (OData) Web API Customer controller. OData is a data-access protocol for the Web that provides a uniform way to query and manipulate data sets through CRUD operations. Using ASP.NET Web API, it’s easy to create an OData endpoint. You can control which OData operations are exposed. You can host multiple OData endpoints alongside non-OData endpoints. You have full control over your data model, back-end business logic and data layer. Figure 6 shows the code for the Customer Web API OData controller.
The code in Figure 6 just creates an OData Web API controller to expose Customer data from the Northwind database. Once this is created, you can run the project, and with tools such as Fiddler (a free Web debugger at fiddler2.com) or LINQPad, you can actually query customer data.
Figure 6 Customer Web API OData Controller
  1. public class CustomerController : EntitySetController<Customer, string>
  2. {
  3.   private readonly NorthwindContext _northwindContext;
  4.   public CustomerController()
  5.   {
  6.     _northwindContext = new NorthwindContext();
  7.   }
  8.   public override IQueryable<Customer> Get()
  9.   {
  10.     return _northwindContext.Customers;
  11.   }
  12.   protected override Customer GetEntityByKey(string key)
  13.   {
  14.     return _northwindContext.Customers.Find(key);
  15.   }
  16.   protected override Customer UpdateEntity(string key, Customer update)
  17.   {
  18.     _northwindContext.Customers.AddOrUpdate(update);
  19.     _northwindContext.SaveChanges();
  20.     return update;
  21.   }
  22.   public override void Delete(string key)
  23.   {
  24.     var customer = _northwindContext.Customers.Find(key);
  25.     _northwindContext.Customers.Remove(customer);
  26.     _northwindContext.SaveChanges();
  27.   }
  28. }
Configuring and Exposing OData from the Customer Table for the Grid (Northwind.Web/App_Start/WebApiConfig.cs)Figure 7 configures and exposes OData from the Customer table for the grid.
Querying OData Web API with LINQPad If you haven’t used LINQPad (linqpad.net) yet, please add this tool to your developer toolkit; it’s a must-have and is available in a free version. Figure 8 shows LINQPad with a connection to the Web API OData (localhost:2501/odata), displaying the results of the LINQ query, “Customer.Take (100).”
Figure 7 Configuring ASP.NET MVC Web API Routes for OData
  1. public static void Register(HttpConfiguration config)
  2. {
  3.   // Web API configuration and services
  4.   ODataModelBuilder modelBuilder = new ODataConventionModelBuilder();
  5.   var customerEntitySetConfiguration =
  6.     modelBuilder.EntitySet<Customer>("Customer");
  7.   customerEntitySetConfiguration.EntityType.Ignore(t => t.Orders);
  8.   customerEntitySetConfiguration.EntityType.Ignore(t =>
  9.      t.CustomerDemographics);
  10.   var model = modelBuilder.GetEdmModel();
  11.   config.Routes.MapODataRoute("ODataRoute""odata", model);
  12.   config.EnableQuerySupport();
  13.   // Web API routes
  14.   config.MapHttpAttributeRoutes();
  15.   config.Routes.MapHttpRoute(
  16.     "DefaultApi""api/{controller}/{id}",
  17.     new {id = RouteParameter.Optional});
  18. }
Querying the Customer Controller Web API OData Via a LINQPad Query
Figure 8 Querying the Customer Controller Web API OData Via a LINQPad Query
Creating the (Observable) Customer Model (Northwind.Web/­Scripts/app/models/customerModel.js) Next is creating the (Kendo UI Observable) Customer model. You can think of this as a client-side Customer entity domain model. I created the Customer model so it can easily be reused by both the Customer grid view and the edit view. The code is shown in Figure 9.
Figure 9 Creating the Customer (Kendo UI Observable) Model
  1. define(['kendo'],
  2.   function (kendo) {
  3.     var customerModel = kendo.data.Model.define({
  4.       id: "CustomerID",
  5.       fields: {
  6.         CustomerID: { type: "string", editable: false, nullable: false },
  7.         CompanyName: { title: "Company", type: "string" },
  8.         ContactName: { title: "Contact", type: "string" },
  9.         ContactTitle: { title: "Title", type: "string" },
  10.         Address: { type: "string" },
  11.         City: { type: "string" },
  12.         PostalCode: { type: "string" },
  13.         Country: { type: "string" },
  14.         Phone: { type: "string" },
  15.         Fax: { type: "string" },
  16.         State: { type: "string" }
  17.       }
  18.     });
  19.     return customerModel;
  20.   });
Creating a DataSource for the Customers Grid (Northwind.Web/Scripts/app/datasources/customersDatasource.js) If you’re familiar with data sources from ASP.NET Web Forms, the concept is the same here, where you create a data source for the Customers grid (Northwind.Web/Scripts/app/datasources/customersDatasource.js). The Kendo UI DataSource (bit.ly/1d0Ycvd) component is an abstraction for using local (arrays of JavaScript objects) or remote (XML, JSON or JSONP) data. It fully supports CRUD data operations and provides both local and server-side support for sorting, paging, filtering, grouping and aggregates.
Creating the View Model for the Customers Grid View If you’re familiar with MVVM from Windows Presentation Foundation (WPF) or Silverlight, this is the same exact concept, just on the client side (found in this project in Northwind.Web/Scripts/ViewModels/­Customer/indexViewModel.cs). MVVM is an architectural separation pattern used to separate the view and its data and business logic. You’ll see in a bit that all the data, business logic and so on is in the view model and that the view is purely HTML (presentation). Figure 10 shows the code for the Customer grid view.
Figure 10 The Customer Grid View Model
  1. define(['kendo''customerDatasource'],
  2.   function (kendo, customerDatasource) {
  3.     var lastSelectedDataItem = null;
  4.     var onClick = function (event, delegate) {
  5.       event.preventDefault();
  6.       var grid = $("#grid").data("kendoGrid");
  7.       var selectedRow = grid.select();
  8.       var dataItem = grid.dataItem(selectedRow);
  9.       if (selectedRow.length > 0)
  10.         delegate(grid, selectedRow, dataItem);
  11.       else
  12.         alert("Please select a row.");
  13.       };
  14.       var indexViewModel = new kendo.data.ObservableObject({
  15.         save: function (event) {
  16.           onClick(event, function (grid) {
  17.             grid.saveRow();
  18.             $(".toolbar").toggle();
  19.           });
  20.         },
  21.         cancel: function (event) {
  22.           onClick(event, function (grid) {
  23.             grid.cancelRow();
  24.             $(".toolbar").toggle();
  25.           });
  26.         },
  27.         details: function (event) {
  28.           onClick(event, function (grid, row, dataItem) {
  29.             router.navigate('/customer/edit/' + dataItem.CustomerID);
  30.           });
  31.         },
  32.         edit: function (event) {
  33.           onClick(event, function (grid, row) {
  34.             grid.editRow(row);
  35.             $(".toolbar").toggle();
  36.           });
  37.         },
  38.         destroy: function (event) {
  39.           onClick(event, function (grid, row, dataItem) {
  40.             grid.dataSource.remove(dataItem);
  41.             grid.dataSource.sync();
  42.           });
  43.         },
  44.         onChange: function (arg) {
  45.           var grid = arg.sender;
  46.           lastSelectedDataItem = grid.dataItem(grid.select());
  47.         },
  48.         dataSource: customerDatasource,
  49.         onDataBound: function (arg) {
  50.           // Check if a row was selected
  51.           if (lastSelectedDataItem == null) return;
  52.           // Get all the rows     
  53.           var view = this.dataSource.view();
  54.           // Iterate through rows
  55.           for (var i = 0; i < view.length; i++) {
  56.           // Find row with the lastSelectedProduct
  57.             if (view[i].CustomerID == lastSelectedDataItem.CustomerID) {
  58.               // Get the grid
  59.               var grid = arg.sender;
  60.               // Set the selected row
  61.               grid.select(grid.table.find("tr[data-uid='" + view[i].uid + "']"));
  62.               break;
  63.             }
  64.           }
  65.         },
  66.       });
  67.       return indexViewModel;
  68.   });
I’ll briefly describe various components of the code in Figure 10:
  • onClick (helper): This method is a helper function, which gets an instance of the Customer grid, the current selected row and a JSON model of the representation of the Customer for the selected row.
  • save: This saves changes when doing an inline edit of a Customer.
  • cancel: This cancels out of inline edit mode.
  • details: This navigates the SPA to the edit Customer view, appending the Customer’s ID to the URL.
  • edit: This activates inline editing for the current selected Customer.
  • destroy: This deletes the current selected Customer.
  • onChange (event): This fires every time a Customer is selected. You store the last selected Customer so you can maintain state. After performing any updates or navigating away from the Customer grid, when navigating back to the grid you reselect the last selected Customer.
Now add customerModel, indexViewModel and customersDatasource modules to your RequireJS configuration (Northwind.Web/Scripts/app/main.js). The code is shown in Figure 11.
Figure 11 RequireJS Configuration Additions
  1. paths: {
  2.   // Packages
  3.   'jquery''/scripts/jquery-2.0.3.min',
  4.   'kendo''/scripts/kendo/2013.3.1119/kendo.web.min',
  5.   'text''/scripts/text',
  6.   'router''/scripts/app/router',
  7.   // Models
  8.   'customerModel''/scripts/app/models/customerModel',
  9.   // View models
  10.   'customer-indexViewModel''/scripts/app/viewmodels/customer/indexViewModel',
  11.   'customer-editViewModel''/scripts/app/viewmodels/customer/editViewModel',
  12.   // Data sources
  13.   'customerDatasource''/scripts/app/datasources/customerDatasource',
  14.   // Utils
  15.   'util''/scripts/util'
  16. }
Add a Route for the New Customers Grid View Note that in the loadView callback (in Northwind.Web/Scripts/app/router.js) you’re binding the toolbar of the grid after it has been initialized and MVVM binding has taken place. This is because the first time you bind your grid, the toolbar hasn’t initialized, because it exists in the grid. When the grid is first initialized via MVVM, it will load in the toolbar from the Kendo UI template. When it’s loaded into the grid, you then bind only the toolbar to your view model so the buttons in your toolbar are bound to the save and cancel methods in your view model. Here’s the relevant code to register the route definition for the Customer edit view:
  1. router.route("/customer/index", function () {
  2.   require(['customer-indexViewModel''text!/customer/index'],
  3.     function (viewModel, view) {
  4.       loadView(viewModel, view, function () {
  5.         kendo.bind($("#grid").find(".k-grid-toolbar"), viewModel);
  6.       });
  7.     });
  8. });
You now have a fully functional Customers grid view. Load up localhost:25061/Home/Spa#/customer/index (the port number will likely vary on your machine) in a browser and you’ll see Figure 12.
The Customer Grid View with MVVM Using the Index View Model
Figure 12 The Customer Grid View with MVVM Using the Index View Model
Wiring Up the Customers Edit View Here are the key steps to add a Customer edit view to the SPA:
  • Create a customer edit view bound to your Customer model via MVVM (Northwind.Web/Views/Customer/Edit.cshtml).
  • Add an edit view model module for the Customer edit view (Northwind.Web/Scripts/app/viewModels/­editViewModel.js).
  • Add a utility helper module to get IDs from the URL (Northwind.Web/Scripts/app/util.js).
Because you’re using the Kendo UI framework, go ahead and style your edit view with Kendo UI styles. You can learn more about that at bit.ly/1f3zWuCFigure 13 shows the edit view markup with an MVVM widget and event binding.
Figure 13 Edit View Markup with an MVVM Widget and Event Binding
  1. <div class="demo-section">
  2.   <div class="k-block" style="padding: 20px">
  3.     <div class="k-block k-info-colored">
  4.       <strong>Note: </strong>Please fill out all of the fields in this form.
  5.     </div>
  6.     <div>
  7.       <dl>
  8.         <dt>
  9.           <label for="companyName">Company Name:</label>
  10.         </dt>
  11.         <dd>
  12.           <input id="companyName" type="text"
  13.             data-bind="value: Customer.CompanyName" class="k-textbox" />
  14.         </dd>
  15.         <dt>
  16.           <label for="contactName">Contact:</label>
  17.         </dt>
  18.         <dd>
  19.           <input id="contactName" type="text"
  20.             data-bind="value: Customer.ContactName" class="k-textbox" />
  21.         </dd>
  22.         <dt>
  23.           <label for="title">Title:</label>
  24.         </dt>
  25.         <dd>
  26.           <input id="title" type="text"
  27.             data-bind="value: Customer.ContactTitle" class="k-textbox" />
  28.         </dd>
  29.         <dt>
  30.           <label for="address">Address:</label>
  31.         </dt>
  32.         <dd>
  33.           <input id="address" type="text"
  34.             data-bind="value: Customer.Address" class="k-textbox" />
  35.         </dd>
  36.         <dt>
  37.           <label for="city">City:</label>
  38.         </dt>
  39.         <dd>
  40.           <input id="city" type="text"
  41.             data-bind="value: Customer.City" class="k-textbox" />
  42.         </dd>
  43.         <dt>
  44.           <label for="zip">Zip:</label>
  45.         </dt>
  46.         <dd>
  47.           <input id="zip" type="text"
  48.             data-bind="value: Customer.PostalCode" class="k-textbox" />
  49.         </dd>
  50.         <dt>
  51.           <label for="country">Country:</label>
  52.         </dt>
  53.         <dd>
  54.           <input id="country" type="text"
  55.           data-bind="value: Customer.Country" class="k-textbox" />
  56.         </dd>
  57.         <dt>
  58.           <label for="phone">Phone:</label>
  59.         </dt>
  60.         <dd>
  61.           <input id="phone" type="text"
  62.             data-bind="value: Customer.Phone" class="k-textbox" />
  63.         </dd>
  64.         <dt>
  65.           <label for="fax">Fax:</label>
  66.         </dt>
  67.         <dd>
  68.           <input id="fax" type="text"
  69.             data-bind="value: Customer.Fax" class="k-textbox" />
  70.         </dd>
  71.       </dl>
  72.       <button data-role="button"
  73.         data-bind="click: saveCustomer"
  74.         data-sprite-css-class="k-icon k-i-tick">Save</button>
  75.       <button data-role="button" data-bind="click: cancel">Cancel</button>
  76.       <style scoped>
  77.         dd
  78.         {
  79.           margin: 0px 0px 20px 0px;
  80.           width: 100%;
  81.         }
  82.         label
  83.         {
  84.           font-size: small;
  85.           font-weight: normal;
  86.         }
  87.         .k-textbox
  88.         {
  89.           width: 100%;
  90.         }
  91.         .k-info-colored
  92.         {
  93.           padding: 10px;
  94.           margin: 10px;
  95.         }
  96.       </style>
  97.     </div>
  98.   </div>
  99. </div>
Create a Utility to Get the ID of the Customer from the URL Because you’re creating concise modules with clean boundaries to create a nice separation of concerns, I’ll demonstrate how to create a Util module where all of your utility helpers will reside. I’ll start with a utility method that can retrieve the customer ID in the URL for the Customer DataSource (Northwind.Web/Scripts/app/datasources/customerDatasource.js), as shown in Figure 14.
Figure 14 The Utility Module
  1. define([],
  2.   function () {
  3.     var util;
  4.     util = {
  5.       getId:
  6.       function () {
  7.         var array = window.location.href.split('/');
  8.         var id = array[array.length - 1];
  9.         return id;
  10.       }
  11.     };
  12.     return util;
  13.   });
Add the Edit View Model and Util Modules to the RequireJS Configuration (Northwind.Web/Scripts/app/main.js) The code in Figure 15 shows RequireJS configuration additions for the Customer edit modules.
Figure 15 RequireJS Configuration Additions for the Customer Edit Modules
  1. require.config({
  2.   paths: {
  3.     // Packages
  4.     'jquery''/scripts/jquery-2.0.3.min',
  5.     'kendo''/scripts/kendo/2013.3.1119/kendo.web.min',
  6.     'text''/scripts/text',
  7.     'router''/scripts/app/router',
  8.     // Models
  9.     'customerModel''/scripts/app/models/customerModel',
  10.     // View models
  11.     'customer-indexViewModel''/scripts/app/viewmodels/customer/indexViewModel',
  12.     'customer-editViewModel''/scripts/app/viewmodels/customer/editViewModel',
  13.     // Data sources
  14.     'customerDatasource''/scripts/app/datasources/customerDatasource',
  15.     // Utils
  16.     'util''/scripts/util'
  17.     },
  18.   shim : {
  19.     'kendo' : ['jquery']
  20.   },
  21.   priority: ['text''router''app'],
  22.   jquery: '2.0.3',
  23.   waitSeconds: 30
  24.   });
  25. require([
  26.   'app'
  27. ], function (app) {
  28.   app.initialize();
  29. });
Add the Customer Edit View Model (Northwind.Web/Scripts/app/viewModels/editViewModel.js)The code in Figure 16 shows how to add a Customer edit view model.
Figure 16 Customer Edit View Model Module for the Customer View
  1. define(['customerDatasource''customerModel''util'],
  2.   function (customerDatasource, customerModel, util) {
  3.     var editViewModel = new kendo.data.ObservableObject({
  4.       loadData: function () {
  5.         var viewModel = new kendo.data.ObservableObject({
  6.           saveCustomer: function (s) {
  7.             customerDatasource.sync();
  8.             customerDatasource.filter({});
  9.             router.navigate('/customer/index');
  10.           },
  11.           cancel: function (s) {
  12.             customerDatasource.filter({});
  13.             router.navigate('/customer/index');
  14.           }
  15.         });
  16.         customerDatasource.filter({
  17.           field: "CustomerID",
  18.           operator: "equals",
  19.           value: util.getId()
  20.         });
  21.         customerDatasource.fetch(function () {
  22.           console.log('editViewModel fetching');
  23.           if (customerDatasource.view().length > 0{
  24.             viewModel.set("Customer", customerDatasource.at(0));
  25.           } else
  26.             viewModel.set("Customer"new customerModel());
  27.         });
  28.         return viewModel;
  29.       },
  30.     });
  31.     return editViewModel;
  32.   });
I’ll briefly describe various components of the code in Figure 16:
  • saveCustomer: This method is responsible for saving any changes on the Customer. It also resets the DataSource’s filter so the grid will be hydrated with all Customers.
  • cancel: This method will navigate the SPA back to the Customer grid view. It also resets the DataSource’s filter so that the grid will be hydrated with all Customers.
  • filter: This invokes the DataSource’s filter method and queries for a specific Customer by the ID that’s in the URL.
  • fetch: This invokes the DataSource’s fetch method after setting up the filter. In the callback of the fetch, you set the Customer property of your view model with the Customer that was returned from your DataSource fetch, which will be used to bind to your Customer edit view.
When RequireJS loads a module, code within the “define” method body will only get invoked once—which is when RequireJS loads the module—so you expose a method (loadData) in your edit view model so you have a mechanism to load data after the edit view model module has already been loaded (see this in Northwind.Web/­Scripts/app/router.js).
Add a Route for the New Customer Edit View (Northwind.Web/Scripts/app/router.js) Here’s the relevant code to add the router:
  1. router.route("/customer/edit/:id",
  2.         function () {
  3.     require(['customer-editViewModel',
  4.           'text!/customer/edit'],
  5.       function (viewModel, view) {
  6.       loadView(viewModel.loadData(), view);
  7.     });
  8.   });
Note that when the Customer edit view model is requested from RequireJS, you’re able to retrieve the Customer by invoking the loadData method from the view model. This way you’re able to load the correct Customer data based on the ID that’s in the URL each and every time the Customer edit view is loaded. A route doesn’t have to be just a hardcoded string. It can also contain parameters, such as a back-end server router (Ruby on Rails, ASP.NET MVC, Django and so on). To do this, you name a route segment with a colon before the variable name you want.
You can now load the Customer edit view in the browser (localhost:25061/Home/Spa#/customer/edit/ANATR) and see the screen depicted in Figure 17.
The Customer Edit View
Figure 17 The Customer Edit View
Note: Although the delete (destroy) functionality on the Customer grid view has been wired up, when clicking the “Delete” button in the toolbar (see Figure 18), you’ll see an exception, as shown in Figure 19.
The Customer Grid View
Figure 18 The Customer Grid View
Expected Exception When Deleting a Customer Due to CustomerID Foreign Key Referential Integrity
Figure 19 Expected Exception When Deleting a Customer Due to CustomerID Foreign Key Referential Integrity
This exception is by design, because most Customer IDs are foreign keys in other tables, for example, Orders, Invoices and so on. You’d have to wire up a cascading delete that would delete all records from all tables where Customer ID is a foreign key. Although you aren’t able to delete anything, I still wanted to show the steps and code for the delete functionality.
So there you have it. I’ve demonstrated how quick and easy it is to convert an out-of-the-box ASP.NET Web Application into an SPA using RequireJS and Kendo UI. Then I showed how easy it is to add CRUD-like functionality to the SPA.
You can see a live demo of the project at bit.ly/1bkMAlK and you can see the CodePlex project site (and downloadable code) at easyspa.codeplex.com.
Happy coding!

Long Le is the principal .NET app/dev architect at CBRE Inc. and a Telerik/Kendo UI MVP. He spends most of his time developing frameworks and application blocks, providing guidance for best practices and patterns and standardizing the enterprise technology stack. He has been working with Microsoft technologies for more than 10 years. In his spare time, he enjoys blogging (blog.longle.net) and playing Call of Duty. You can reach and follow him on Twitter at twitter.com/LeLong37.
Thanks to the following technical experts for reviewing this article: Derick Bailey (Telerik) and Mike Wasson (Microsoft)

Nhận xét

  1. Are you trying to earn cash from your visitors by using popunder ads?
    If so, did you ever use ExoClick?

    Trả lờiXóa

Đăng nhận xét