Part 2: Filter/Search using Angular2 pipe, Global Error handling, Debugging Client sidehtml
Part 3: Angular 2 to Angular 4 with Angular Material UI Componentsnode
In this article, we will continue enhancing the User Management application by upgrading the Angular version from Angular 2 to Angular 4 and using Angular Material 2 components instead of traditional HTML and third-party components (e.g. ng2-bs3-modal modal pop up we used for Add/Update/Delete User).react
Angular 4 has couple of new cool features, I don’t want to go over them since you can easily find them online. Check here for quick review. We will use Email Validator
and If-else
Template conditions in User Management application.git
Angular Material is Angular compatible components that we mostly use to design the web application’s UI e.g. Input
, Autocomplete
, dropdown
, checkbox
etc. Click here to review the Angular Material components. We will replace all current HTML and third-party components with the Angular Material components.github
This article is the third part of Angular2 in ASP.NET MVC & Web API - Part 2. In previous articles, we used ng3-bs3-modal
third-party modal pop up components and traditional HTML controls. In this article, we will be replacing all controls with Angular Material components. It is highly recommended that you go through Part 1 & Part 2 before reading this article.web
Before starting the actual development, let me show what would be the final output of this article. You can compare it with Angular2 in ASP.NET MVC & Web API - Part 2 to get an idea what we are going to build.chrome
You can see, we added few more form controls on Add User
modal pop up and look & feel of each control is very different from Part 1 & Part 2. Let’s start the development:typescript
As this article is continuation of Angular2 in ASP.NET MVC & Web API - Part 2 article, let’s download the attached project from here.npm
Angular2MVC_p2.sln
in Visual Studio 2017 Community Edition (Preferably), it is also recommended to rename the solution.package.json
file and replace the file content with the following packages:{
"name": "angular-quickstart", "version": "1.0.0", "description": "QuickStart package.json from the documentation for visual studio 2017 & WebApi", "scripts": { "start": "tsc && concurrently \"tsc -w\" \"lite-server\" ", "lint": "tslint ./app/**/*.ts -t verbose", "lite": "lite-server", "pree2e": "webdriver-manager update", "test": "tsc && concurrently \"tsc -w\" \"karma start karma.conf.js\"", "test-once": "tsc && karma start karma.conf.js --single-run", "tsc": "tsc", "tsc:w": "tsc -w" }, "keywords": [], "author": "", "license": "MIT", "dependencies": { "@angular/common": "4.0.2", "@angular/compiler": "4.0.2", "@angular/core": "4.0.2", "@angular/forms": "4.0.2", "@angular/http": "4.0.2", "@angular/platform-browser": "4.0.2", "@angular/platform-browser-dynamic": "4.0.2", "@angular/router": "4.0.2", "angular-in-memory-web-api": "0.2.4", "systemjs": "0.19.40", "core-js": "2.4.1", "rxjs": "5.0.1", "zone.js": "0.7.4" }, "devDependencies": { "concurrently": "3.2.0", "lite-server": "2.2.2", "typescript": "2.0.10", "canonical-path": "0.0.2", "tslint": "3.15.1", "lodash": "4.16.4", "jasmine-core": "2.4.1", "karma": "1.3.0", "karma-chrome-launcher": "2.0.0", "karma-cli": "1.0.1", "karma-jasmine": "1.0.2", "karma-jasmine-html-reporter": "0.2.2", "protractor": "4.0.14", "rimraf": "2.5.4", "@types/node": "6.0.46", "@types/jasmine": "2.5.36", "@angular/material": "2.0.0-beta.6", "@angular/animations": "4.1.3" }, "repository": {} }
dependencies
section, we are upgrading the Angular & other helping packages version to 4.0.2
.devDependencies
section, you can see we are importing the Angular Material package:package.json
file and select option Restore Packages
, it will take few minutes to download all the package. Wait until you get package restore complete message.Build -> Rebuild Solution
option. It will take few seconds or minute to download all packages (.Net & Angular). That’s it with Angular 4 packages restore.User
table, let’s add some more to better understand the Angular Material components.App_Data
folder, right click and select Open
or double click on UserDB.mdf
file to edit it:Tables
from Data Connections -> UserDBEntities
hierarchy, right click on TblUser
and select option Open Table Defination
:Id
, FirstName
, LastName
, Gender
), let's manually update the table according to the following screen shot:Update
button. It will take few moments and you will end up with the following screen, click on Update Database
button:User
entity, go to DBContext
folder, right click on UserDBEntities.edmx
and select option Open
or double click to edit it:Update Model from Database
…Update Wizard
screen, go to Refresh
tab, select Tables
and click on Finish
button:TblUser
would be updated as follow:We are all set with Database update, let move to next steps.
Let’s update our User Management application to use Angular Material components and few features of Angular 4.
app -> app.module.ts
and add following import statements for Angular Material. Also add the BrowserAnimationsModule
, MaterialModule
, MdNativeDateModule
modules in import section:<span id="ArticleContent">import { BrowserAnimationsModule } from '@angular/platform-browser/animations'; import { MaterialModule, MdNativeDateModule } from '@angular/material';</span>
systemjs.config.js
file.<span id="ArticleContent">'@angular/animations': 'npm:@angular/animations/bundles/animations.umd.js', '@angular/animations/browser': 'npm:@angular/animations/bundles/animations-browser.umd.js', '@angular/platform-browser/animations': 'npm:@angular/platform-browser/bundles/platform-browser-animations.umd.js', '@angular/material': 'npm:@angular/material/bundles/material.umd.js', </span>
AppModule
, our application is ready to use Angular Material components. First let’s modify the home
page. Edit the app -> Components -> home.component.ts
file.Card
from Angular Material, go to Angular Material Card page to understand what is Card, where and how to use it?Card
page and for any other component, you can see there are three tabs OVERVIEW
, API REFERENCE
and EXAMPLE
that give us following information:
OVERVIEW
: Component description, architecture and it’s use. HTML, TS and CSS code and link to working Plunker.API REFERENCE
: How to import component, Module and Directive information.EXAMPLE
: Running example with almost all feature of specific component and Plunker link with HTML, TS and CSS code.Example
Tab then click on View Source < >
link.HTML
, TS
and CSS
.HTML
, Typescript
(if required any) and CSS
from here to our application for all components where we would need them. Of course, we will modify it according to our requirements but DO NOT shy to copy, this really help us to take full advantage of Angular Material components without any effort. So, copy the entire md-card
HTML and replace it with home.component.ts
template’s HTML:CSS
for many components, so let’s create stylesheet. Right click on Content
folder and select Add -> Style Sheet
:style.css
and click on OK
button:style.css
content with following CSS
:<span id="ArticleContent">.example-card { width: 400px; } .example-header-image { background-image: url('../images/users.png'); background-size: cover; }</span>
App_Start -> BundleConfig.cs
and update the Content/css
bundle as follow to add and use the newly added style.css
in our application:<span id="ArticleContent">bundles.Add(new StyleBundle("~/Content/css").Include( "~/Content/style.css", "~/Content/bootstrap.css"));</span>
home.component.ts
and modify it. Change the md-card-title
e.g. "Users", md-card-subtitle
as "Sample Image" etc. Update the image URL in md-card-image
to src="../../images/users.png"
. The final home.component.ts
template should look like following:user.component.ts
and update it with the Angular Material components. To make it simple and use Dialog component, we need to break it in two components. The UserComponent
will only have User list whereas Add
, Update
and Delete
functionality will be moved to ManageUser
component that we going to create in next steps.ng2-bs3-modal
modal pop up control for add, update and delete screen that is not needed any more, so remove all of its references from app.module.ts
, systemjs.config.js
and user.component.ts
files.ManageUser
component and then we will come back to UserComponent
to clean it. Right click on app -> Components
and select Add -> TypeScript
file:manageuser.component.ts
and click on OK
button. Also create the manageuser.component.html
file:User
table with few additional columns, let’s update our user.ts
accordingly. Edit the app-> Model -> user.ts
and update it as following:<span id="ArticleContent">export interface IUser { Id: number, FirstName: string, LastName: string, Email:string, Gender: string, DOB: string, City: string, State: string, Zip: string, Country:string }</span>
manageuser.component.html
, we will add Angular Material Dialog. We will simply go to Dialog Plunker and copy the dialog-result-example-dialog.html
file content to manageuser.component.html
. From this file, you can see there are three main section of Dialog, title
, content
and action buttons
that is quite self-explanatory. The other Angular Material components correspond to user.ts
model fields that will go inside md-dialog-content
div are as follow:
Plunker
link for each component to see running example, I personally prefer Plunker to get the code for each component.manageuser.component.html
file:<span id="ArticleContent"><form novalidate (ngSubmit)="onSubmit(userFrm)" [formGroup]="userFrm"> <div> <h1 md-dialog-title><span><md-icon>create</md-icon>{{modalTitle}}</span></h1> </div> <div style="padding-bottom:1px; padding: 0px; border: 0px; color: rgb(0, 0, 255);">></div> <div md-dialog-content class="md-dialog-container"> <div class="frm-ctrl"> <md-input-container> <input mdInput placeholder="First Name" formControlName="FirstName"> </md-input-container> <div *ngIf="formErrors.FirstName" class="text-danger"> {{ formErrors.FirstName }} </div> </div> <div class="frm-ctrl"> <md-input-container> <input mdInput placeholder="Last Name" formControlName="LastName"> </md-input-container> <div *ngIf="formErrors.LastName" class="text-danger"> {{ formErrors.LastName }} </div> </div> <div class="frm-ctrl"> <md-input-container> <input type="email" mdInput placeholder="Email" formControlName="Email"> </md-input-container> <div *ngIf="formErrors.Email" class="text-danger"> {{ formErrors.Email }} </div> </div> <div class="frm-ctrl"> <md-radio-group formControlName="Gender"> <md-radio-button *ngFor="let gndr of gender" [value]="gndr"> {{gndr}} </md-radio-button> </md-radio-group> <div *ngIf="formErrors.Gender" class="text-danger"> {{ formErrors.Gender }} </div> </div> <div class