Create the Angular Application
Create Project
- Create Project Folder named SuperHeroNG
data:image/s3,"s3://crabby-images/765c3/765c37f4092ddeba7b6890c22fab01a50bbf22d1" alt=""
- Download Node.js
data:image/s3,"s3://crabby-images/c7bf8/c7bf81dcb29cc9a9c873ec21c540129239edf0a7" alt=""
- Open project forder in Visual Studio Code
- Set powershell policy with
Set-ExecutionPolicy -Scope CurrentUser -ExecutionPolicy RemoteSigned
- Download Angular CLI with
npm install -g @angular/cli
- Create new angular projekt skeleton with
ng new SuperHero.UI
in SuperHeroNG folder
data:image/s3,"s3://crabby-images/032f9/032f9c1513f8af9afe3ce0f46c912d7b9b10ea31" alt=""
data:image/s3,"s3://crabby-images/28407/28407de0fc0ccd420e0bf1e9798d6c2b25d89e27" alt=""
data:image/s3,"s3://crabby-images/f0e1c/f0e1c935857d43621665ae5d1aed874a634293b4" alt=""
- Add
components
,models
andservices
folders in src/app - Run angular with
ng serve -o
in SuperHero.UI folder
data:image/s3,"s3://crabby-images/e82e4/e82e47bc853b1d0144934a41adfd0dc80b9129ec" alt=""
data:image/s3,"s3://crabby-images/6bb78/6bb78fca83c11cf196bc485510b680c17a3b96b0" alt=""
Add the Super Hero Model to the Client
Download Extensions
- Download
Angular Language Service
in Visual Studio Code
data:image/s3,"s3://crabby-images/0c6ae/0c6ae28d5223560c9bd0e7877022db9b62ffa5d4" alt=""
Super Hero Model
- SuperHero.UI/src/app/models/super-hero.ts
export class SuperHero {
id?: number;
name = "";
firstName = "";
lastName = "";
place = "";
}
Generate an Angular Service
Service
- Move to app\services
- Create super-hero service with
ng g s super-hero --skip-tests
data:image/s3,"s3://crabby-images/06347/06347eab0007b0efc85f1a72cbb03f1e0e0c2c30" alt=""
- SuperHero.UI/src/app/services/super-hero.service.ts
...
export class SuperHeroService {
constructor() { }
public getSuperHeroes() : SuperHero[] {
let hero = new SuperHero();
hero.id = 1;
hero.name = "Spider Man";
hero.firstName = "Peter";
hero.lastName = "Parker";
hero.place = "New York City";
return [hero];
}
...
Component
-
Clear app/app.component.html
-
SuperHero.UI/src/app/app.component.ts
...
export class AppComponent {
title = 'SuperHero.UI';
heroes: SuperHero[] = [];
constructor(private superHeroService: SuperHeroService) {}
ngOnInit() : void {
this.heroes = this.superHeroService.getSuperHeroes();
console.log(this.heroes);
}
...
Run
- Run with
ng serve -o
data:image/s3,"s3://crabby-images/e9d62/e9d6296a5a9f217da3d2a71f74d70cf8b6deeded" alt=""
data:image/s3,"s3://crabby-images/da141/da14138104a4463b245f0cb35536a7a0c02feb96" alt=""
Implement a Table to Show the Hero List
- SuperHero.UI/src/app/app.component.html
<table>
<thead>
<th>Name</th>
<th>First Name</th>
<th>Last Name</th>
<th>Place</th>
</thead>
<tbody>
<tr *ngFor="let hero of heroes">
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
data:image/s3,"s3://crabby-images/70386/7038659c163c033da3841ea047c641480e7e52b9" alt=""
Create the .Net 6 Web API Project
- Create a ASP.NET CORE WEB API project named SuperHeroAPI on same folder with UI in Visual Studio
data:image/s3,"s3://crabby-images/34b55/34b559b9abc0f76355eea84d9f81aa7f2d8cbc9f" alt=""
Add the Super Hero Model to the API
- SuperHeroAPI/SuperHero.cs
public class SuperHero
{
public int Id { get; set; }
public string Name { get; set; } = string.Empty;
public string FirstName { get; set; } = string.Empty;
public string LastName { get; set; } = string.Empty;
public string Place { get; set; } = string.Empty;
}
Add the Super Hero Controller
- SuperHeroAPI/Controller/SuperHeroController.cs
[Route("api/[controller]")]
[ApiController]
public class SuperHeroController : ControllerBase
{
[HttpGet]
public async Task<ActionResult<List<SuperHero>>> GetSuperHeroes()
{
return new List<SuperHero>
{
new SuperHero
{
Name = "Spider Man",
FirstName = "Peter",
LastName = "Parker",
Place = "New York City"
}
};
}
}
Call the Web API from the Angular 14 Client
-
Copy your Web API url and paste on your angular.
-
SuperHero.UI/src/environment/environment.ts
export const environment = {
production: false,
apiUrl: "https://localhost:7169/api"
};
- SuperHero.UI/src/app/services/super-hero.service.ts
...
export class SuperHeroService {
private url = "SuperHero";
constructor(private http: HttpClient) { }
public getSuperHeroes() : Observable<SuperHero[]> {
return this.http.get<SuperHero[]>(`${environment.apiUrl}/${this.url}`);
}
}
...
- SuperHero.UI/src/app/app.module.ts
...
import { HttpClientModule } from '@angular/common/http';
@NgModule({
declarations: [
AppComponent
],
imports: [
BrowserModule,
AppRoutingModule,
HttpClientModule
],
...
- SuperHero.UI/src/app/components/app.component.ts
...
ngOnInit() : void {
this.superHeroService
.getSuperHeroes()
.subscribe((result: SuperHero[]) => (this.heroes = result));
}
...
Enable CORS(Cross-Origin Resource Sharing)
- SuperHero.API/Program.cs
builder.Services.AddCors(options => options.AddPolicy(name: "SuperHeroOrigins", // To Access the Origins
policy =>
{
policy.WithOrigins("http://localhost:4200").AllowAnyMethod().AllowAnyHeader();
}));
...
app.UseCors("SuperHeroOrigins");
...
data:image/s3,"s3://crabby-images/7fb25/7fb25bd4757092bad1a50acbcb75f046638a5bf9" alt=""
Add Entity Framework Core 6
-
Create Data folder in SuperHeroAPI
-
SuperHero.API/Data/DataContext.cs
public class DataContext : DbContext
{
public DataContext(DbContextOptions<DataContext> options) : base(options)
{
}
public DbSet<SuperHero> SuperHeroes => Set<SuperHero>();
}
- Download
Microsoft.EntityFrameworkCore.Design
andMicrosoft.EntityFrameworkCore.SqlServer
in Nuget Package manager
data:image/s3,"s3://crabby-images/a85ed/a85eda4318c7aed4dc870d06fb4c214a81176fb4" alt=""
Add the Connection String for the SQL Server Database
- SuperHero.API/appsettings.json
...
"ConnectionStrings": {
"DefaultConnection": "server=localhost\\sqlexpress;database=superherodb;trusted_connection=true"
},
...
Install the EF Core Tools
- Download Dotnet EntityCore Tools with
dotnet tool install --global dotnet-ef
and check withdotnet ef
data:image/s3,"s3://crabby-images/a9a49/a9a495ba4ffe075d267d12e0c1b64083345ca167" alt=""
data:image/s3,"s3://crabby-images/a5529/a55292296d24e04481418d4abc8491afc944e557" alt=""
Register the DataContext in the Program.cs
- SuperHero.API/Program.cs
builder.Services.AddDbContext<DataContext>(options =>
{
options.UseSqlServer(builder.Configuration.GetConnectionString("DefaultConnection"));
});
Run the Initial Migration using Code-First Migration
- Migration data table with
dotnet ef migrations add Initial
data:image/s3,"s3://crabby-images/3d092/3d092f0476945df8a12e0f8dd198a00a8139c3d8" alt=""
- Update database with
dotnet ef database update
data:image/s3,"s3://crabby-images/96bd5/96bd5651dc163ec428198321f5fcda696df34741" alt=""
Get Super Heroes from the Database
- SuperHero.API/Controllers/SuperHeroController.cs
[Route("api/[controller]")]
[ApiController]
public class SuperHeroController : ControllerBase
{
private readonly DataContext _context;
public SuperHeroController(DataContext context)
{
_context = context;
}
[HttpGet]
public async Task<ActionResult<List<SuperHero>>> GetSuperHeroes()
{
return Ok(await _context.SuperHeroes.ToListAsync());
}
}
- Add a row in database
data:image/s3,"s3://crabby-images/96bd5/96bd5651dc163ec428198321f5fcda696df34741" alt=""
Implement Create, Update and Delete in the SuperHeroController
- SuperHero.API/Controllers/SuperHeroController.cs
...
[HttpPost]
public async Task<ActionResult<List<SuperHero>>> CreateSuperHero(SuperHero hero)
{
_context.SuperHeroes.Add(hero);
await _context.SaveChangesAsync();
return Ok(await _context.SuperHeroes.ToListAsync());
}
[HttpPut]
public async Task<ActionResult<List<SuperHero>>> UpdateSuperHero(SuperHero hero)
{
var dbHero = await _context.SuperHeroes.FindAsync(hero.Id);
if (dbHero == null)
return BadRequest("Hero not found.");
dbHero.Name = hero.Name;
dbHero.FirstName = hero.FirstName;
dbHero.LastName = hero.LastName;
dbHero.Place = hero.Place;
await _context.SaveChangesAsync();
return Ok(await _context.SuperHeroes.ToListAsync());
}
[HttpDelete("{id}")]
public async Task<ActionResult<List<SuperHero>>> DeleteSuperHero(int id)
{
var dbHero = await _context.SuperHeroes.FindAsync(id);
if (dbHero == null)
return BadRequest("Hero not found.");
_context.SuperHeroes.Remove(dbHero);
await _context.SaveChangesAsync();
return Ok(await _context.SuperHeroes.ToListAsync());
}
...
Build a Edit-Hero Component with Angular
- Add a new component with
ng n c edit-hero --skip-tests
in app\components
data:image/s3,"s3://crabby-images/be3a7/be3a70c4aaa50086523b3d6ca62f0fc2cd481e93" alt=""
- SuperHero.UI/src/app/components/edit-hero.component.ts
...
export class EditHeroComponent implements OnInit {
@Input() hero?: SuperHero;
...
updateHero(hero:SuperHero){
}
deleteHero(hero:SuperHero){
}
createHero(hero:SuperHero){
}
...
- SuperHero.UI/src/app.module.ts
...
imports: [
BrowserModule,
AppRoutingModule,
HttpClientModule,
FormsModule
],
...
- SuperHero.UI/src/app/components/edit-hero.component.ts
<div *ngIf="hero">
<h2></h2>
<div>
Name:
<input [(ngModel)]="hero.name" placeholder="Name"/>
</div>
<div>
First Name:
<input [(ngModel)]="hero.firstName" placeholder="First Name"/>
</div>
<div>
Last Name:
<input [(ngModel)]="hero.lastName" placeholder="Last Name"/>
</div>
<div>
Place:
<input [(ngModel)]="hero.place" placeholder="Place"/>
</div>
<button (click)="updateHero(hero)" *ngIf="hero.id">Save</button>
<button (click)="deleteHero(hero)" *ngIf="hero.id">Delete</button>
<button (click)="createHero(hero)" *ngIf="!hero.id">Create</button>
</div>
Add the Edit-Hero Form to the Parent Form
- SuperHero.UI/src/app/app.component.html
<button (click)="initNewHero()">Create New Hero</button>
<table>
<thead>
<th>Name</th>
<th>First Name</th>
<th>Last Name</th>
<th>Place</th>
</thead>
<tbody>
<tr *ngFor="let hero of heroes">
<td></td>
<td></td>
<td></td>
<td></td>
<td><button (click)="editHero(hero)">Edit</button></td>
</tr>
</tbody>
</table>
<app-edit-hero [hero]="heroToEdit"></app-edit-hero>
- SuperHero.UI/src/app/app.component.ts
export class AppComponent {
title = 'SuperHero.UI';
heroes: SuperHero[] = [];
heroToEdit?: SuperHero;
constructor(private superHeroService: SuperHeroService) {}
ngOnInit() : void {
this.superHeroService
.getSuperHeroes()
.subscribe((result: SuperHero[]) => (this.heroes = result));
}
initNewHero(){
this.heroToEdit = new SuperHero();
}
editHero(hero: SuperHero){
this.heroToEdit = hero;
}
}
data:image/s3,"s3://crabby-images/eecd4/eecd4d4e9fb3827f45e40fd121b4c79ef5a88c00" alt=""
Implement Web Service Calls on the client
- SuperHero.UI/src/app/services/super-hero.service.ts
export class SuperHeroService {
private url = "SuperHero";
constructor(private http: HttpClient) { }
public getSuperHeroes() : Observable<SuperHero[]> {
return this.http.get<SuperHero[]>(`${environment.apiUrl}/${this.url}`);
}
public upgradeHero(hero: SuperHero) : Observable<SuperHero[]> {
return this.http.put<SuperHero[]>(`${environment.apiUrl}/${this.url}`, hero);
}
public createHero(hero: SuperHero) : Observable<SuperHero[]> {
return this.http.post<SuperHero[]>(`${environment.apiUrl}/${this.url}`, hero);
}
public deleteHero(hero: SuperHero) : Observable<SuperHero[]> {
return this.http.delete<SuperHero[]>(`${environment.apiUrl}/${this.url}/${hero.id}`);
}
}
Call the Service in the EditHero Component
- SuperHero.UI/src/app/components/edit-hero.component.ts
export class EditHeroComponent implements OnInit {
@Input() hero?: SuperHero;
@Output() heroesUpdated = new EventEmitter<SuperHero[]>();
constructor(private superHeroService: SuperHeroService) { }
ngOnInit(): void {}
updateHero(hero:SuperHero){
this.superHeroService.upgradeHero(hero).subscribe((heroes: SuperHero[]) => this.heroesUpdated.emit(heroes));
}
deleteHero(hero:SuperHero){
this.superHeroService.deleteHero(hero).subscribe((heroes: SuperHero[]) => this.heroesUpdated.emit(heroes));
}
createHero(hero:SuperHero){
this.superHeroService.createHero(hero).subscribe((heroes: SuperHero[]) => this.heroesUpdated.emit(heroes));
}
}
Update the Super Hero Table
- SuperHero.UI/src/app/app.component.ts
export class AppComponent {
...
updateHeroList(heroes: SuperHero[]){
this.heroes = heroes;
}
...
- SuperHero.UI/src/app/app.component.html
...
<app-edit-hero [hero]="heroToEdit" (heroesUpdated)="updateHeroList($event)"></app-edit-hero>
UI
Read
data:image/s3,"s3://crabby-images/143a1/143a12f476c803f82d216db4ed806f2e9256c318" alt=""
Create
data:image/s3,"s3://crabby-images/a2184/a21848fc4b157ac3544e7e3000a28af66bc6f3cc" alt=""
data:image/s3,"s3://crabby-images/e11aa/e11aab75f66441ec5cb16affa47b2d1150bebe3b" alt=""
Update
data:image/s3,"s3://crabby-images/3b66d/3b66d865025be3a3187ed70a75228fc6624e8d00" alt=""
Delete
data:image/s3,"s3://crabby-images/c8264/c8264f6e26da2eec5c46f68a6e74bbf9dcd8a230" alt=""