WIP
Some checks failed
Quality Check / Linting (push) Failing after 20s

This commit is contained in:
Dominik Säume 2025-01-09 10:44:02 +01:00
parent a336d856e1
commit 79e5df8ddb
5 changed files with 226 additions and 6 deletions

View file

@ -1,4 +1,10 @@
import {Routes} from '@angular/router'; import { Routes } from '@angular/router';
import {DashboardComponent} from '@app/views/dashboard/dashboard.component'; import { DashboardComponent } from '@app/views/dashboard/dashboard.component';
export const routes: Routes = [{path: '', component: DashboardComponent, title: 'Home'}]; import { EmployeeDetailComponent } from './views/employee-detail/employee-detail.component';
export const routes: Routes = [
{ path: '', component: DashboardComponent, title: 'Home' },
{ path: 'employee/new', component: EmployeeDetailComponent, title: 'New Employee' },
{ path: 'employee/:id', component: EmployeeDetailComponent, title: 'Edit Employee' }
];

View file

@ -0,0 +1,68 @@
<form class="employee-detail" [formGroup]="employeeForm" appearance="outline">
<div class="employee-detail__info-view">
<div class="employee-detail__info-view__base">
<h2>Info</h2>
<mat-form-field>
<mat-label>First Name</mat-label>
<input matInput formControlName="firstName">
</mat-form-field>
<mat-form-field>
<mat-label>Last Name</mat-label>
<input matInput formControlName="lastName">
</mat-form-field>
<mat-form-field>
<mat-label>Postcode</mat-label>
<mat-icon matSuffix>money</mat-icon>
<input matInput formControlName="postcode">
</mat-form-field>
<mat-form-field>
<mat-label>City</mat-label>
<mat-icon matSuffix>location_city</mat-icon>
<input matInput formControlName="city">
</mat-form-field>
<mat-form-field>
<mat-label>Street</mat-label>
<mat-icon matSuffix>house</mat-icon>
<input matInput formControlName="street">
</mat-form-field>
<mat-form-field>
<mat-label>Phone</mat-label>
<mat-icon matSuffix>phone</mat-icon>
<input matInput formControlName="phone">
</mat-form-field>
</div>
<div class="employee-detail__info-view__skills">
<h2>Skills</h2>
<mat-table [dataSource]="skillsDataSource">
<ng-container matColumnDef="id">
<th mat-header-cell *matHeaderCellDef>Id</th>
<td mat-cell *matCellDef="let skill">{{skill.id}}</td>
</ng-container>
<ng-container matColumnDef="skill">
<th mat-header-cell *matHeaderCellDef>Skill</th>
<td mat-cell *matCellDef="let skill">{{skill.skill}}</td>
</ng-container>
<ng-container matColumnDef="action">
<th mat-header-cell *matHeaderCellDef></th>
<td mat-cell *matCellDef="let row">
<button mat-mini-fab class="warn shadowless">
<mat-icon>delete</mat-icon>
</button>
</td>
</ng-container>
<tr mat-row *matRowDef="let row; columns: skillsDisplayedColumns;"></tr>
</mat-table>
</div>
</div>
<div class="employee-detail__action-row">
<a mat-flat-button class="abort" routerLink="/">Abort</a>
@if (isNewEmployee) {
<button mat-flat-button (click)="onCreate()">Create</button>
}
@else {
<button mat-flat-button (click)="onSave()">Save</button>
}
</div>
</form>

View file

@ -0,0 +1,60 @@
@use '@angular/material' as mat;
.employee-detail {
display: flex;
flex-direction: column;
gap: 1rem;
&__info-view {
display: flex;
gap: 2rem;
>* {
min-width: 0;
flex-basis: 0;
flex-grow: 1;
}
&__base {
display: flex;
flex-direction: column;
input {
font-size: var(--mat-sys-body-large-size);
}
}
&__skills {
height: fit-content;
width: 100%;
tr {
// Match the Spacing of the Formular to the left
font-size: var(--mat-sys-body-large-size);
height: 76px;
td {
width: 100%;
padding-top: 10px;
}
.mat-column-action {
width: 0;
}
&:first-of-type {
height: 66px;
td {
padding-top: 0;
}
}
}
}
}
&__action-row {
display: flex;
justify-content: space-between;
}
}

View file

@ -0,0 +1,85 @@
import { Component } from '@angular/core';
import { FormBuilder, FormGroup, ReactiveFormsModule } from '@angular/forms';
import { MatButton, MatButtonModule, MatMiniFabButton } from '@angular/material/button';
import { MatFormField, MatLabel, MatSuffix } from '@angular/material/form-field';
import { MatIcon } from '@angular/material/icon';
import { MatInput } from '@angular/material/input';
import { MatCell, MatCellDef, MatColumnDef, MatHeaderCell, MatHeaderCellDef, MatRow, MatRowDef, MatTable, MatTableDataSource } from '@angular/material/table';
import { ActivatedRoute, RouterLink } from '@angular/router';
import { Employee, EmployeeService, Qualification } from '@core/ems';
@Component({
selector: 'app-employee-detail',
imports: [
ReactiveFormsModule,
MatFormField,
MatLabel,
MatInput,
MatIcon,
MatSuffix,
MatTable,
MatRowDef,
MatColumnDef,
MatHeaderCellDef,
MatCell,
MatCellDef,
MatHeaderCell,
MatRow,
MatMiniFabButton,
RouterLink,
MatButtonModule
],
templateUrl: './employee-detail.component.html',
styleUrl: './employee-detail.component.scss'
})
export class EmployeeDetailComponent {
isNewEmployee: boolean;
employeeForm: FormGroup;
skillsDataSource: MatTableDataSource<Qualification> = new MatTableDataSource<Qualification>([]);
skillsDisplayedColumns = ['skill', 'action'];
constructor(
private route: ActivatedRoute,
private employeeService: EmployeeService,
private formBuilder: FormBuilder
) {
const idParam = this.route.snapshot.paramMap.get('id');
this.isNewEmployee = idParam == null;
this.employeeForm = this.formBuilder.group<Employee>({
id: 0,
firstName: '',
lastName: '',
postcode: '',
city: '',
street: '',
phone: '',
skillSet: []
});
if (this.isNewEmployee) {
return;
}
const id = parseInt(idParam as string);
this.employeeService.getEmployee({ id }).subscribe((employee) => {
this.employeeForm = this.formBuilder.group<Employee>(employee);
this.skillsDataSource = new MatTableDataSource<Qualification>(employee.skillSet);
});
}
onAbort() {
}
onCreate() {
}
onSave() {
}
}

View file

@ -29,7 +29,8 @@ app-root {
align-items: center; align-items: center;
} }
.mdc-button { .mdc-button,
.mdc-fab {
&.abort, &.abort,
&.error, &.error,
&.warn { &.warn {
@ -52,12 +53,12 @@ app-root {
)); ));
@include mat.fab-overrides(( @include mat.fab-overrides((
// default // default
container-color: $fg-container, container-color: $bg-container,
foreground-color: $fg-container, foreground-color: $fg-container,
state-layer-color: $fg-container, state-layer-color: $fg-container,
ripple-color: $ripple, ripple-color: $ripple,
// mini // mini
small-container-color: $fg-container, small-container-color: $bg-container,
small-foreground-color: $fg-container, small-foreground-color: $fg-container,
small-state-layer-color: $fg-container, small-state-layer-color: $fg-container,
small-ripple-color: $ripple small-ripple-color: $ripple