Skip to content

Latest commit

 

History

History
211 lines (154 loc) · 8.33 KB

README.md

File metadata and controls

211 lines (154 loc) · 8.33 KB

ASP.NET MVC With Angular 2+

How to add Angular 2+ (NOT angular.js so) to an existing ASP.NET MVC With Razor or Aspx View engine project for having a hybrid front-end.

Made with MVC5 but the principle is the same with Core if server side rendering is used.

  • How to integrate the Angular app
  • How to make WEP API calls with the same authentication token and anti-forgery
  • How to manage the navigation
  • How to exchange data by javascript

run the POC

Software pre-requites (tested on win10 - x64)

Installation:

  • Get the repo
  • Open the solution -> manage NuGet packages -> restore
  • Navigate with command prompt to the AngularApp folder into the project -> npm install
  • after npm install from the same folder in command prompt: npm run-script build-watch
  • In Visual Studio -> start debbuging

If problem with database creation (Cannot attach the file and so on...) Open the package manager console in VS:

> Add-migration repair
> Update-database

Emergency only:)

> sqllocaldb.exe stop 
> sqllocaldb.exe delete 
> sqllocaldb.exe start 
> Add-migration repair
> Update-database

stack

.NET

  • .NET Framework 4.6.1
  • ASP.NET MVC 5.2.4
  • Razor (works with forms as well)
  • Forms authentication (just an example, even better with openId)
  • ASP.NET Identity (just an example)

Angular

  • Angular CLI: 6.0.7
  • Node: 8.11.2

How to reproduce

We considere the .Net project is an existing one. Actually for the POC, it's a new MVC Project with forms authentication .

pre-requites: Angular cli

Create a new Angular project

Navigate with command prompt to your MVC WEB project root path and run

	ng new AngularApp

Or whatever name you want. The created folder will be included into your project, just think about that.

Add the essentials following files to your project:

  • AngularApp/Src folder
  • AngularApp/package.json
  • AngularApp/angular.json
  • AngularApp/tsconfig.json
  • AngularApp/node_modules

Update configuration settings:

  • angular.json: change the outputPath to make it pointing to ../Scripts/angular folder as part of your MVC 5 project
	...,
	"build": {
          "builder": "@angular-devkit/build-angular:browser",
          "options": {
            "outputPath": "../Scripts/angular",
			...

If you want to use the same ASP.NET layout as your MVC app, add <base href="/"> on its header. It could be done in angular by programming too but be carreful, it will be new variable to set for deployment.

Create a new view in your project. This view will be the host of the angular app. In my POC, just another view named Angular linked to the Home controller with the same layout as the web app.

Add the javascript links for the angular js files (another constrain to manage for deployment: theses files names) and the app-root tag (It tells Angular where our angular components should be rendered)

Complete HTML of the Angular.cshtml view

Add an action to your MVC controler for this view, a link to the view in your layout for example and that's it!

Navigation

How to navigate from your .Net part to your angular part? It's really not complicate but think about one thing: everytime you want to display a .Net page with angular on it, you bootstrap the angular root module to launch the application. It means angular will start everytime you are displaying a page in .Net with the angular js files on it.

As a simple example, we will create a navigation in angular to show how the hybrid works (page1, page2, page3 and nav)

Create the 4 angular components: Navigate with command prompt to your Angular root folder path (AngularApp for me) and run

  ng g component page1
  ng g component page2
  ng g component page3
  ng g component nav

Your new components are created in AngularApp/src/app

In AngularApp/src, create the new file app.routes.ts app.routes.ts

In this POC, my .Net manage /Home/Angular to display my MVC view which will bootstrap Angular. All the other asks after /Home/Angular/ (e.g. /Home/Angular/link1, /Home/Angular/etc/1) will be managed by angular because the Angular router is loaded from it.

In AngularApp/src/app.module.ts, Import the RouterModule and your ROUTES file app.module.ts

import { RouterModule } from '@angular/router';
import { ROUTES } from './app.routes';

@NgModule({
  ...,
  imports: [
    ...,
    RouterModule.forRoot(ROUTES)
  ],
  ...
})

Change the AngularApp/src/app/nav/nav.component.html HTML content nav.component.html On each page component (e.g. /AngularApp/src/app/page1/page1.component.html) add the <app-nav></app-nav> tag for having the nav component on each of them

Note:

  • routerLink: navigate to Angular page
  • HREF: navigate to .Net page

if HREF point to an .Net managed URL where ANgular is present, Angular will load and take the url for itself

You can test and and see you can navigate from .Net to Angular, Angular to .Net, navigate into Angular and even ask a refresh on the page where ANgular is present: it will manage the URL as well. Of course there is a boostrap and you must be aware about that.

WEB APIs and Forms authentication

As the Angular host page is a .Net page, this page is protected with the same level as the others. However, when you will use the Angular app, you have to implement your authorization layer on it for displaying a link or a button, based on the user role for example. I wont speak about that here, it's really pure Angular implementation. What is interesting for now is to be sure our frontend is safe enough for speaking with our backend thanks to the WEB APIs.

Add the following controller for the POC: new WEB API 2 Controller with read/write actions

We will protect this controller with a simple anti-forgery pattern (I'm not certain about this code, you will just have the operating principle for that). Create a class ValidateHttpAntiForgeryTokenAttribute.cs and copy the following code on it

Now you can decorate your get with AUthorize and ValidateHttpAntiForgeryToken. See the complete code here

    // GET: api/Angular
    [Authorize]
    [ValidateHttpAntiForgeryToken]
    public IEnumerable<string> Get()
    {
       ...

Now we will call this WEB API controller from Angular.

Be sure you have the @Html.AntiForgeryToken() on your page which host the angular app Complete HTML of the Angular.cshtml view

On your main Angular module, add the HttpClientModule module. app.module.ts

import { HttpClientModule } from '@angular/common/http';

@NgModule({
  ...
  imports: [
      HttpClientModule,
  ],
})

principle: we will get the RequestVerificationToken value and add it into the request header. For example on page2, change the code in the typescript file for making the call and in the html file for displying the result.

javascript context exchange

It's a quick and really durty solution to use if really you need to exchange public js data from .Net to Angular.

Thanks for the help: