Erster und wichtigster Push :D
16
software/Frontend/.browserslistrc
Normal file
@ -0,0 +1,16 @@
|
||||
# This file is used by the build system to adjust CSS and JS output to support the specified browsers below.
|
||||
# For additional information regarding the format and rule options, please see:
|
||||
# https://github.com/browserslist/browserslist#queries
|
||||
|
||||
# For the full list of supported browsers by the Angular framework, please see:
|
||||
# https://angular.io/guide/browser-support
|
||||
|
||||
# You can see what browsers were selected by your queries by running:
|
||||
# npx browserslist
|
||||
|
||||
last 1 Chrome version
|
||||
last 1 Firefox version
|
||||
last 2 Edge major versions
|
||||
last 2 Safari major versions
|
||||
last 2 iOS major versions
|
||||
Firefox ESR
|
16
software/Frontend/.editorconfig
Normal file
@ -0,0 +1,16 @@
|
||||
# Editor configuration, see https://editorconfig.org
|
||||
root = true
|
||||
|
||||
[*]
|
||||
charset = utf-8
|
||||
indent_style = space
|
||||
indent_size = 2
|
||||
insert_final_newline = true
|
||||
trim_trailing_whitespace = true
|
||||
|
||||
[*.ts]
|
||||
quote_type = single
|
||||
|
||||
[*.md]
|
||||
max_line_length = off
|
||||
trim_trailing_whitespace = false
|
42
software/Frontend/.gitignore
vendored
Normal file
@ -0,0 +1,42 @@
|
||||
# See http://help.github.com/ignore-files/ for more about ignoring files.
|
||||
|
||||
# Compiled output
|
||||
/dist
|
||||
/tmp
|
||||
/out-tsc
|
||||
/bazel-out
|
||||
|
||||
# Node
|
||||
/node_modules
|
||||
npm-debug.log
|
||||
yarn-error.log
|
||||
|
||||
# IDEs and editors
|
||||
.idea/
|
||||
.project
|
||||
.classpath
|
||||
.c9/
|
||||
*.launch
|
||||
.settings/
|
||||
*.sublime-workspace
|
||||
|
||||
# Visual Studio Code
|
||||
.vscode/*
|
||||
!.vscode/settings.json
|
||||
!.vscode/tasks.json
|
||||
!.vscode/launch.json
|
||||
!.vscode/extensions.json
|
||||
.history/*
|
||||
|
||||
# Miscellaneous
|
||||
/.angular/cache
|
||||
.sass-cache/
|
||||
/connect.lock
|
||||
/coverage
|
||||
/libpeerconnection.log
|
||||
testem.log
|
||||
/typings
|
||||
|
||||
# System files
|
||||
.DS_Store
|
||||
Thumbs.db
|
27
software/Frontend/README.md
Normal file
@ -0,0 +1,27 @@
|
||||
# Frontend
|
||||
|
||||
This project was generated with [Angular CLI](https://github.com/angular/angular-cli) version 14.2.8.
|
||||
|
||||
## Development server
|
||||
|
||||
Run `ng serve` for a dev server. Navigate to `http://localhost:4200/`. The application will automatically reload if you change any of the source files.
|
||||
|
||||
## Code scaffolding
|
||||
|
||||
Run `ng generate component component-name` to generate a new component. You can also use `ng generate directive|pipe|service|class|guard|interface|enum|module`.
|
||||
|
||||
## Build
|
||||
|
||||
Run `ng build` to build the project. The build artifacts will be stored in the `dist/` directory.
|
||||
|
||||
## Running unit tests
|
||||
|
||||
Run `ng test` to execute the unit tests via [Karma](https://karma-runner.github.io).
|
||||
|
||||
## Running end-to-end tests
|
||||
|
||||
Run `ng e2e` to execute the end-to-end tests via a platform of your choice. To use this command, you need to first add a package that implements end-to-end testing capabilities.
|
||||
|
||||
## Further help
|
||||
|
||||
To get more help on the Angular CLI use `ng help` or go check out the [Angular CLI Overview and Command Reference](https://angular.io/cli) page.
|
105
software/Frontend/angular.json
Normal file
@ -0,0 +1,105 @@
|
||||
{
|
||||
"$schema": "./node_modules/@angular/cli/lib/config/schema.json",
|
||||
"version": 1,
|
||||
"newProjectRoot": "projects",
|
||||
"projects": {
|
||||
"Frontend": {
|
||||
"projectType": "application",
|
||||
"schematics": {},
|
||||
"root": "",
|
||||
"sourceRoot": "src",
|
||||
"prefix": "app",
|
||||
"architect": {
|
||||
"build": {
|
||||
"builder": "@angular-devkit/build-angular:browser",
|
||||
"options": {
|
||||
"outputPath": "dist/frontend",
|
||||
"index": "src/index.html",
|
||||
"main": "src/main.ts",
|
||||
"polyfills": "src/polyfills.ts",
|
||||
"tsConfig": "tsconfig.app.json",
|
||||
"assets": [
|
||||
"src/favicon.ico",
|
||||
"src/assets"
|
||||
],
|
||||
"styles": [
|
||||
"src/custom-theme.scss",
|
||||
"src/styles.css",
|
||||
"node_modules/leaflet/dist/leaflet.css"
|
||||
],
|
||||
"scripts": []
|
||||
},
|
||||
"configurations": {
|
||||
"production": {
|
||||
"budgets": [
|
||||
{
|
||||
"type": "initial",
|
||||
"maximumWarning": "500kb",
|
||||
"maximumError": "1mb"
|
||||
},
|
||||
{
|
||||
"type": "anyComponentStyle",
|
||||
"maximumWarning": "2kb",
|
||||
"maximumError": "4kb"
|
||||
}
|
||||
],
|
||||
"fileReplacements": [
|
||||
{
|
||||
"replace": "src/environments/environment.ts",
|
||||
"with": "src/environments/environment.prod.ts"
|
||||
}
|
||||
],
|
||||
"outputHashing": "all"
|
||||
},
|
||||
"development": {
|
||||
"buildOptimizer": false,
|
||||
"optimization": false,
|
||||
"vendorChunk": true,
|
||||
"extractLicenses": false,
|
||||
"sourceMap": true,
|
||||
"namedChunks": true
|
||||
}
|
||||
},
|
||||
"defaultConfiguration": "production"
|
||||
},
|
||||
"serve": {
|
||||
"builder": "@angular-devkit/build-angular:dev-server",
|
||||
"configurations": {
|
||||
"production": {
|
||||
"browserTarget": "Frontend:build:production"
|
||||
},
|
||||
"development": {
|
||||
"browserTarget": "Frontend:build:development"
|
||||
}
|
||||
},
|
||||
"defaultConfiguration": "development"
|
||||
},
|
||||
"extract-i18n": {
|
||||
"builder": "@angular-devkit/build-angular:extract-i18n",
|
||||
"options": {
|
||||
"browserTarget": "Frontend:build"
|
||||
}
|
||||
},
|
||||
"test": {
|
||||
"builder": "@angular-devkit/build-angular:karma",
|
||||
"options": {
|
||||
"main": "src/test.ts",
|
||||
"polyfills": "src/polyfills.ts",
|
||||
"tsConfig": "tsconfig.spec.json",
|
||||
"karmaConfig": "karma.conf.js",
|
||||
"assets": [
|
||||
"src/favicon.ico",
|
||||
"src/assets"
|
||||
],
|
||||
"styles": [
|
||||
"src/custom-theme.scss",
|
||||
"src/styles.css",
|
||||
"node_modules/leaflet/dist/leaflet.css"
|
||||
],
|
||||
"scripts": []
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
44
software/Frontend/karma.conf.js
Normal file
@ -0,0 +1,44 @@
|
||||
// Karma configuration file, see link for more information
|
||||
// https://karma-runner.github.io/1.0/config/configuration-file.html
|
||||
|
||||
module.exports = function (config) {
|
||||
config.set({
|
||||
basePath: '',
|
||||
frameworks: ['jasmine', '@angular-devkit/build-angular'],
|
||||
plugins: [
|
||||
require('karma-jasmine'),
|
||||
require('karma-chrome-launcher'),
|
||||
require('karma-jasmine-html-reporter'),
|
||||
require('karma-coverage'),
|
||||
require('@angular-devkit/build-angular/plugins/karma')
|
||||
],
|
||||
client: {
|
||||
jasmine: {
|
||||
// you can add configuration options for Jasmine here
|
||||
// the possible options are listed at https://jasmine.github.io/api/edge/Configuration.html
|
||||
// for example, you can disable the random execution with `random: false`
|
||||
// or set a specific seed with `seed: 4321`
|
||||
},
|
||||
clearContext: false // leave Jasmine Spec Runner output visible in browser
|
||||
},
|
||||
jasmineHtmlReporter: {
|
||||
suppressAll: true // removes the duplicated traces
|
||||
},
|
||||
coverageReporter: {
|
||||
dir: require('path').join(__dirname, './coverage/frontend'),
|
||||
subdir: '.',
|
||||
reporters: [
|
||||
{ type: 'html' },
|
||||
{ type: 'text-summary' }
|
||||
]
|
||||
},
|
||||
reporters: ['progress', 'kjhtml'],
|
||||
port: 9876,
|
||||
colors: true,
|
||||
logLevel: config.LOG_INFO,
|
||||
autoWatch: true,
|
||||
browsers: ['Chrome'],
|
||||
singleRun: false,
|
||||
restartOnFileChange: true
|
||||
});
|
||||
};
|
21168
software/Frontend/package-lock.json
generated
Normal file
50
software/Frontend/package.json
Normal file
@ -0,0 +1,50 @@
|
||||
{
|
||||
"name": "frontend",
|
||||
"version": "0.0.0",
|
||||
"scripts": {
|
||||
"ng": "ng",
|
||||
"start": "ng serve",
|
||||
"build": "ng build",
|
||||
"watch": "ng build --watch --configuration development",
|
||||
"test": "ng test"
|
||||
},
|
||||
"private": true,
|
||||
"dependencies": {
|
||||
"@angular/animations": "^14.2.0",
|
||||
"@angular/cdk": "^13.0.0",
|
||||
"@angular/common": "^14.2.0",
|
||||
"@angular/compiler": "^14.2.0",
|
||||
"@angular/core": "^14.2.0",
|
||||
"@angular/forms": "^14.2.0",
|
||||
"@angular/material": "^13.0.0",
|
||||
"@angular/platform-browser": "^14.2.0",
|
||||
"@angular/platform-browser-dynamic": "^14.2.0",
|
||||
"@angular/router": "^14.2.0",
|
||||
"@types/googlemaps": "^3.43.3",
|
||||
"@types/leaflet": "^1.9.3",
|
||||
"leaflet": "^1.9.3",
|
||||
"mqtt": "^4.3.7",
|
||||
"ngx-mqtt": "^15.0.0-alpha.1",
|
||||
"rxjs": "~7.5.0",
|
||||
"tslib": "^2.3.0",
|
||||
"zone.js": "~0.11.4"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@angular-devkit/build-angular": "^14.2.8",
|
||||
"@angular/cli": "~14.2.8",
|
||||
"@angular/compiler-cli": "^14.2.0",
|
||||
"@types/jasmine": "~4.0.0",
|
||||
"jasmine-core": "~4.3.0",
|
||||
"karma": "~6.4.0",
|
||||
"karma-chrome-launcher": "~3.1.0",
|
||||
"karma-coverage": "~2.2.0",
|
||||
"karma-jasmine": "~5.1.0",
|
||||
"karma-jasmine-html-reporter": "~2.0.0",
|
||||
"typescript": "~4.7.2"
|
||||
},
|
||||
"description": "This project was generated with [Angular CLI](https://github.com/angular/angular-cli) version 14.2.8.",
|
||||
"main": "karma.conf.js",
|
||||
"keywords": [],
|
||||
"author": "",
|
||||
"license": "ISC"
|
||||
}
|
@ -0,0 +1 @@
|
||||
Subproject commit a13b150dfe70b0beb1484f051ad4d7076ba7fd62
|
55
software/Frontend/src/app/Home/home.component.css
Normal file
@ -0,0 +1,55 @@
|
||||
#robotIsOnTour {
|
||||
color: red;
|
||||
font-size: larger;
|
||||
}
|
||||
|
||||
#robotIsReady {
|
||||
color: green;
|
||||
font-size: larger;
|
||||
}
|
||||
|
||||
.container {
|
||||
display: grid;
|
||||
grid-template-rows: 40% 20% 7%;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
font-family: Cormorant;
|
||||
}
|
||||
|
||||
.item-input {
|
||||
grid-row: 1 / 2;
|
||||
grid-column: 1/-1;
|
||||
justify-self: center;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
.item-box-input {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.item-box-output {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.item-output {
|
||||
grid-row: 3 / 4;
|
||||
grid-column: 1/-1;
|
||||
justify-self: start;
|
||||
align-self: start;
|
||||
|
||||
}
|
||||
|
||||
|
||||
.item-robotText {
|
||||
grid-row: 2 / 3;
|
||||
grid-column: 1/-1;
|
||||
justify-self: center;
|
||||
align-self: center;
|
||||
font-size: 1.2rem;
|
||||
padding: 0.7rem;
|
||||
border: 0.125rem;
|
||||
border-style: solid;
|
||||
border-radius: 1rem;
|
||||
border-color: silver;
|
||||
box-shadow: 0.2rem 0.2rem 0.2rem silver;
|
||||
}
|
16
software/Frontend/src/app/Home/home.component.html
Normal file
@ -0,0 +1,16 @@
|
||||
<div class="container">
|
||||
<div class="item-box-input">
|
||||
<div class="item-input">
|
||||
<app-input></app-input>
|
||||
</div>
|
||||
</div>
|
||||
<div class="item-box-output">
|
||||
<div class="item-output">
|
||||
<app-output></app-output>
|
||||
</div>
|
||||
</div>
|
||||
<div class="item-robotText">
|
||||
<div *ngIf="robotStateChange$ | async" id="robotIsOnTour">Robot is on Tour</div>
|
||||
<div *ngIf="!(robotStateChange$ | async)" id="robotIsReady">Robot is Ready</div>
|
||||
</div>
|
||||
</div>
|
23
software/Frontend/src/app/Home/home.component.spec.ts
Normal file
@ -0,0 +1,23 @@
|
||||
// import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { HomeComponent } from './home.component';
|
||||
|
||||
// describe('HomeComponent', () => {
|
||||
// let component: HomeComponent;
|
||||
// let fixture: ComponentFixture<HomeComponent>;
|
||||
|
||||
// beforeEach(async () => {
|
||||
// await TestBed.configureTestingModule({
|
||||
// declarations: [ HomeComponent ]
|
||||
// })
|
||||
// .compileComponents();
|
||||
|
||||
// fixture = TestBed.createComponent(HomeComponent);
|
||||
// component = fixture.componentInstance;
|
||||
// fixture.detectChanges();
|
||||
// });
|
||||
|
||||
// it('should create', () => {
|
||||
// expect(component).toBeTruthy();
|
||||
// });
|
||||
// });
|
16
software/Frontend/src/app/Home/home.component.ts
Normal file
@ -0,0 +1,16 @@
|
||||
import { Component } from '@angular/core';
|
||||
import { Observable } from 'rxjs';
|
||||
import { StoreService } from '../Service/store.service';
|
||||
|
||||
@Component({
|
||||
selector: 'app-home',
|
||||
templateUrl: './home.component.html',
|
||||
styleUrls: ['./home.component.css']
|
||||
})
|
||||
export class HomeComponent {
|
||||
robotStateChange$: Observable<boolean> = this.storeService.currentRobotReady;
|
||||
|
||||
constructor(private storeService: StoreService) {
|
||||
}
|
||||
|
||||
}
|
39
software/Frontend/src/app/Home/input/input.component.css
Normal file
@ -0,0 +1,39 @@
|
||||
.container {
|
||||
display: grid;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
grid-template-rows: 25% 75%;
|
||||
font-family: Cormorant;
|
||||
margin-top: 3rem;
|
||||
}
|
||||
|
||||
.inputText {
|
||||
grid-row: 1 / 2;
|
||||
justify-self: center;
|
||||
align-self: center;
|
||||
font-size: 2rem;
|
||||
}
|
||||
|
||||
.inputMicro {
|
||||
width: 8.3rem;
|
||||
height: 100%;
|
||||
grid-row: 2 / 3;
|
||||
justify-self: center;
|
||||
align-self: start;
|
||||
}
|
||||
|
||||
button {
|
||||
padding: 6.25rem 2.31rem;
|
||||
border: 0.125rem;
|
||||
border-style: solid;
|
||||
border-radius: 10.625rem;
|
||||
border-color: silver;
|
||||
box-shadow: 0.2rem 0.2rem 0.2rem silver;
|
||||
background-color: white;
|
||||
|
||||
}
|
||||
|
||||
mat-icon {
|
||||
transform: scale(8);
|
||||
margin-left: 1rem;
|
||||
}
|
10
software/Frontend/src/app/Home/input/input.component.html
Normal file
@ -0,0 +1,10 @@
|
||||
<div class="container">
|
||||
<div class="inputText">
|
||||
<span>{{text}}</span>
|
||||
</div>
|
||||
<div class="inputMicro">
|
||||
<button mat-menu-item (click)="startStopService()">
|
||||
<mat-icon [ngStyle]="{'color': micIsClicked ? 'blue' : 'silver' }">micro</mat-icon>
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
69
software/Frontend/src/app/Home/input/input.component.spec.ts
Normal file
@ -0,0 +1,69 @@
|
||||
import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
import { MqttRequestService } from 'src/app/Service/Mqtt/mqtt-request.service';
|
||||
import { SpeechService } from '../speech.service';
|
||||
import { InputComponent } from './input.component';
|
||||
|
||||
describe('InputComponent', () => {
|
||||
let component: InputComponent;
|
||||
let fixture: ComponentFixture<InputComponent>;
|
||||
let speechService: SpeechService;
|
||||
let mqttRequestService: MqttRequestService;
|
||||
|
||||
beforeEach(async () => {
|
||||
await TestBed.configureTestingModule({
|
||||
declarations: [InputComponent],
|
||||
providers: [SpeechService, MqttRequestService],
|
||||
}).compileComponents();
|
||||
});
|
||||
|
||||
beforeEach(() => {
|
||||
fixture = TestBed.createComponent(InputComponent);
|
||||
component = fixture.componentInstance;
|
||||
speechService = TestBed.inject(SpeechService);
|
||||
mqttRequestService = TestBed.inject(MqttRequestService);
|
||||
fixture.detectChanges();
|
||||
});
|
||||
|
||||
it('should create the InputComponent', () => {
|
||||
expect(component).toBeTruthy();
|
||||
});
|
||||
|
||||
it('should initialize the SpeechService if not already initialized', () => {
|
||||
spyOn(speechService, 'init');
|
||||
component.ngOnInit();
|
||||
expect(speechService.init).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should update the text and publish to INPUT when receiving a message', () => {
|
||||
const message = 'Test message';
|
||||
spyOn(mqttRequestService, 'publishToINPUT');
|
||||
component.ngOnInit();
|
||||
// component.service.currentApprovalStageMessage.next(message);
|
||||
expect(component.text).toBe(message);
|
||||
expect(mqttRequestService.publishToINPUT).toHaveBeenCalledWith(message);
|
||||
});
|
||||
|
||||
it('should stop the service when micIsClicked is true', () => {
|
||||
spyOn(component.service, 'stop');
|
||||
component.micIsClicked = true;
|
||||
component.startStopService();
|
||||
expect(component.service.stop).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should start the service when micIsClicked is false', () => {
|
||||
spyOn(component.service, 'start');
|
||||
component.micIsClicked = false;
|
||||
component.startStopService();
|
||||
expect(component.service.start).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should unsubscribe from subscription and call onDestroy', () => {
|
||||
spyOn(component.subscription, 'unsubscribe');
|
||||
spyOn(component.service, 'onDestory');
|
||||
component.ngOnDestroy();
|
||||
expect(component.subscription.unsubscribe).toHaveBeenCalled();
|
||||
expect(component.service.onDestory).toHaveBeenCalled();
|
||||
expect(component.text).toBe('');
|
||||
});
|
||||
});
|
||||
|
51
software/Frontend/src/app/Home/input/input.component.ts
Normal file
@ -0,0 +1,51 @@
|
||||
import { Component, OnDestroy, OnInit } from '@angular/core';
|
||||
import { Subscription } from 'rxjs-compat';
|
||||
import { MqttRequestService } from 'src/app/Service/Mqtt/mqtt-request.service';
|
||||
import { SpeechService } from '../speech.service';
|
||||
|
||||
|
||||
@Component({
|
||||
selector: 'app-input',
|
||||
templateUrl: './input.component.html',
|
||||
styleUrls: ['./input.component.css']
|
||||
})
|
||||
export class InputComponent implements OnInit, OnDestroy {
|
||||
text: string = 'Init';
|
||||
micIsClicked = false;
|
||||
subscription: Subscription;
|
||||
|
||||
constructor(
|
||||
public service: SpeechService,
|
||||
private mqttRequestService: MqttRequestService
|
||||
) {
|
||||
if (!this.service.initSpeechServiceStarted) {
|
||||
this.service.init();
|
||||
}
|
||||
}
|
||||
|
||||
ngOnInit(): void {
|
||||
this.subscription = this.service.currentApprovalStageMessage.subscribe(msg => {
|
||||
this.text = msg;
|
||||
if (this.text !== '') {
|
||||
this.micIsClicked = false;
|
||||
this.mqttRequestService.publishToINPUT(this.text);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
startStopService() {
|
||||
if (this.micIsClicked) {
|
||||
this.micIsClicked = false;
|
||||
this.service.stop();
|
||||
} else {
|
||||
this.micIsClicked = true;
|
||||
this.service.start();
|
||||
}
|
||||
}
|
||||
ngOnDestroy() {
|
||||
this.subscription.unsubscribe()
|
||||
this.service.onDestory();
|
||||
this.text = '';
|
||||
}
|
||||
|
||||
}
|
34
software/Frontend/src/app/Home/output/output.component.css
Normal file
@ -0,0 +1,34 @@
|
||||
.container {
|
||||
display: grid;
|
||||
grid-template-columns: 10% 90%;
|
||||
height: 100%;
|
||||
width: 100%;
|
||||
font-family: Cormorant;
|
||||
|
||||
border: 0.125rem;
|
||||
border-style: solid;
|
||||
border-radius: 1rem;
|
||||
border-color: silver;
|
||||
box-shadow: 0.2rem 0.2rem 0.2rem silver;
|
||||
}
|
||||
|
||||
.outputIcon {
|
||||
grid-column: 1 / 2;
|
||||
justify-self: start;
|
||||
align-self: end;
|
||||
|
||||
}
|
||||
|
||||
.outputText {
|
||||
grid-column: 2 / 3;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
font-size: 2rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.mat-icon {
|
||||
margin-left: 0.5rem;
|
||||
margin-top: 0.5rem;
|
||||
transform: scale(1.5);
|
||||
}
|
@ -0,0 +1,8 @@
|
||||
<div class="container">
|
||||
<div class="outputIcon">
|
||||
<mat-icon>volume_up</mat-icon>
|
||||
</div>
|
||||
<div class="outputText">
|
||||
<div>{{outputText}}</div>
|
||||
</div>
|
||||
</div>
|
@ -0,0 +1,23 @@
|
||||
// import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { OutputComponent } from './output.component';
|
||||
|
||||
// describe('OutputComponent', () => {
|
||||
// let component: OutputComponent;
|
||||
// let fixture: ComponentFixture<OutputComponent>;
|
||||
|
||||
// beforeEach(async () => {
|
||||
// await TestBed.configureTestingModule({
|
||||
// declarations: [ OutputComponent ]
|
||||
// })
|
||||
// .compileComponents();
|
||||
|
||||
// fixture = TestBed.createComponent(OutputComponent);
|
||||
// component = fixture.componentInstance;
|
||||
// fixture.detectChanges();
|
||||
// });
|
||||
|
||||
// it('should create', () => {
|
||||
// expect(component).toBeTruthy();
|
||||
// });
|
||||
// });
|
38
software/Frontend/src/app/Home/output/output.component.ts
Normal file
@ -0,0 +1,38 @@
|
||||
import { Component, OnDestroy, OnInit } from '@angular/core';
|
||||
import { Subscription } from 'rxjs';
|
||||
|
||||
import { MqttRequestService } from 'src/app/Service/Mqtt/mqtt-request.service';
|
||||
|
||||
declare var speechSynthesis: any;
|
||||
|
||||
@Component({
|
||||
selector: 'app-output',
|
||||
templateUrl: './output.component.html',
|
||||
styleUrls: ['./output.component.css']
|
||||
})
|
||||
export class OutputComponent implements OnInit, OnDestroy {
|
||||
subscription: Subscription;
|
||||
outputText: string = '';
|
||||
constructor(private mqttRequestService: MqttRequestService) { }
|
||||
|
||||
ngOnInit(): void {
|
||||
this.subscription = this.mqttRequestService.responseMessage.subscribe(msg => {
|
||||
setTimeout(() => {
|
||||
this.outputText = msg;
|
||||
this.speak(this.outputText);
|
||||
}, 3000);
|
||||
});
|
||||
}
|
||||
|
||||
speak(text: string) {
|
||||
const utterance = new SpeechSynthesisUtterance(text);
|
||||
utterance.lang = 'en-US';
|
||||
speechSynthesis.speak(utterance);
|
||||
}
|
||||
|
||||
ngOnDestroy() {
|
||||
this.subscription.unsubscribe();
|
||||
this.mqttRequestService.onDestroy();
|
||||
}
|
||||
|
||||
}
|
16
software/Frontend/src/app/Home/speech.service.spec.ts
Normal file
@ -0,0 +1,16 @@
|
||||
// import { TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { SpeechService } from './speech.service';
|
||||
|
||||
// describe('SpeechService', () => {
|
||||
// let service: SpeechService;
|
||||
|
||||
// beforeEach(() => {
|
||||
// TestBed.configureTestingModule({});
|
||||
// service = TestBed.inject(SpeechService);
|
||||
// });
|
||||
|
||||
// it('should be created', () => {
|
||||
// expect(service).toBeTruthy();
|
||||
// });
|
||||
// });
|
75
software/Frontend/src/app/Home/speech.service.ts
Normal file
@ -0,0 +1,75 @@
|
||||
import { Injectable } from '@angular/core';
|
||||
import { BehaviorSubject } from 'rxjs-compat';
|
||||
declare var webkitSpeechRecognition: any;
|
||||
|
||||
|
||||
@Injectable({
|
||||
providedIn: 'root'
|
||||
})
|
||||
export class SpeechService {
|
||||
private approvalStageMessage = new BehaviorSubject('');
|
||||
currentApprovalStageMessage = this.approvalStageMessage.asObservable();
|
||||
initSpeechServiceStarted: boolean = false;
|
||||
|
||||
recognition = new webkitSpeechRecognition();
|
||||
|
||||
isStoppedSpeechRecog = false;
|
||||
public text = '';
|
||||
tempWords: string | undefined;
|
||||
|
||||
constructor() {
|
||||
this.recognition.pitch = 10;
|
||||
this.recognition.rate = 3;
|
||||
}
|
||||
|
||||
init() {
|
||||
this.initSpeechServiceStarted = true;
|
||||
this.recognition.interimResults = true;
|
||||
this.recognition.lang = 'en-US';
|
||||
|
||||
this.recognition.addEventListener('result', (e: { results: Iterable<unknown> | ArrayLike<unknown>; }) => {
|
||||
const transcript = Array.from(e.results)
|
||||
.map((result: any) => result[0])
|
||||
.map((result) => result.transcript)
|
||||
.join('');
|
||||
this.tempWords = transcript;
|
||||
console.log('init:' + transcript);
|
||||
});
|
||||
}
|
||||
|
||||
start() {
|
||||
this.text = '';
|
||||
this.isStoppedSpeechRecog = false;
|
||||
this.recognition.start();
|
||||
console.log("Speech recognition started")
|
||||
this.recognition.addEventListener('end', () => {
|
||||
this.wordConcat()
|
||||
this.recognition.stop();
|
||||
if (!this.isStoppedSpeechRecog) {
|
||||
this.isStoppedSpeechRecog = true;
|
||||
console.log("End speech recognition");
|
||||
this.approvalStageMessage.next(this.text);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
stop() {
|
||||
this.recognition.stop();
|
||||
if (!this.isStoppedSpeechRecog) {
|
||||
this.isStoppedSpeechRecog = true;
|
||||
console.log("End speech recognition");
|
||||
this.approvalStageMessage.next(this.text);
|
||||
}
|
||||
}
|
||||
|
||||
wordConcat() {
|
||||
this.text = this.text + ' ' + this.tempWords;
|
||||
this.tempWords = '';
|
||||
}
|
||||
|
||||
onDestory() {
|
||||
this.text = '';
|
||||
this.approvalStageMessage.next(this.text);
|
||||
this.isStoppedSpeechRecog = false;
|
||||
}
|
||||
}
|
5
software/Frontend/src/app/Map/map.component.css
Normal file
@ -0,0 +1,5 @@
|
||||
div {
|
||||
height: 70%;
|
||||
width: 80%;
|
||||
margin: 2% 9%;
|
||||
}
|
1
software/Frontend/src/app/Map/map.component.html
Normal file
@ -0,0 +1 @@
|
||||
<div #mapContainer></div>
|
23
software/Frontend/src/app/Map/map.component.spec.ts
Normal file
@ -0,0 +1,23 @@
|
||||
// import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { MapComponent } from './map.component';
|
||||
|
||||
// describe('MapComponent', () => {
|
||||
// let component: MapComponent;
|
||||
// let fixture: ComponentFixture<MapComponent>;
|
||||
|
||||
// beforeEach(async () => {
|
||||
// await TestBed.configureTestingModule({
|
||||
// declarations: [MapComponent]
|
||||
// })
|
||||
// .compileComponents();
|
||||
|
||||
// fixture = TestBed.createComponent(MapComponent);
|
||||
// component = fixture.componentInstance;
|
||||
// fixture.detectChanges();
|
||||
// });
|
||||
|
||||
// it('should create', () => {
|
||||
// expect(component).toBeTruthy();
|
||||
// });
|
||||
// });
|
23
software/Frontend/src/app/Map/map.component.ts
Normal file
@ -0,0 +1,23 @@
|
||||
import { AfterViewInit, Component, ElementRef, ViewChild } from '@angular/core';
|
||||
import * as L from 'leaflet';
|
||||
import 'leaflet/dist/leaflet.css';
|
||||
|
||||
@Component({
|
||||
selector: 'app-map',
|
||||
templateUrl: './map.component.html',
|
||||
styleUrls: ['./map.component.css']
|
||||
})
|
||||
export class MapComponent implements AfterViewInit {
|
||||
@ViewChild('mapContainer', { static: false }) mapContainer: ElementRef;
|
||||
|
||||
ngAfterViewInit(): void {
|
||||
const map = L.map(this.mapContainer.nativeElement).setView([51.505, -0.09], 13);
|
||||
|
||||
L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {
|
||||
}).addTo(map);
|
||||
|
||||
L.marker([49.452785, 11.096963]).addTo(map) //Koordinaten [X,Y]
|
||||
.bindPopup('Robot')
|
||||
.openPopup();
|
||||
}
|
||||
}
|
@ -0,0 +1,85 @@
|
||||
.IMG-Plants {
|
||||
border-radius: 0.03125rem;
|
||||
width: 9.25rem;
|
||||
height: 7.375rem;
|
||||
margin: 0rem 0.3125rem;
|
||||
}
|
||||
|
||||
.icons-measurements {
|
||||
width: 1rem;
|
||||
height: 0.75rem;
|
||||
margin: 0rem 0.3125rem;
|
||||
}
|
||||
|
||||
.container {
|
||||
min-width: 40rem;
|
||||
width: 100%;
|
||||
min-height: 9rem;
|
||||
padding: 0.3125rem;
|
||||
box-shadow: 0.3125rem 0.3125rem 0.3125rem silver;
|
||||
border-radius: 1.25rem;
|
||||
border-color: silver;
|
||||
border-width: 0.0625rem;
|
||||
border-style: solid;
|
||||
margin-bottom: 1rem;
|
||||
font-family: Cormorant;
|
||||
display: grid;
|
||||
grid-template-columns: 45% 50% 5%;
|
||||
justify-content: center;
|
||||
align-content: center;
|
||||
font-size: 1.125rem;
|
||||
}
|
||||
|
||||
.item-header-content {
|
||||
grid-column: 1/2;
|
||||
justify-self: center;
|
||||
align-self: center;
|
||||
display: grid;
|
||||
width: 100%;
|
||||
grid-template-columns: 55% 45%;
|
||||
grid-template-rows: 25% 75%;
|
||||
justify-content: start;
|
||||
align-content: start;
|
||||
}
|
||||
|
||||
.item-header {
|
||||
grid-column: 1/3;
|
||||
grid-row: 1/2;
|
||||
justify-self: center;
|
||||
align-self: start;
|
||||
font-size: 1.875rem;
|
||||
margin-bottom: 1rem;
|
||||
}
|
||||
|
||||
.item-Measurements-IconsAndNames {
|
||||
grid-column: 1/2;
|
||||
grid-row: 2/3;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
.item-Measurements-Values {
|
||||
grid-column: 2/3;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
.item-IMG {
|
||||
justify-self: center;
|
||||
align-self: end;
|
||||
}
|
||||
|
||||
.item-delete {
|
||||
justify-self: end;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
|
||||
.delete-icon {
|
||||
transform: scale(1.2);
|
||||
}
|
||||
|
||||
|
||||
p {
|
||||
margin: 0;
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
<div class="container">
|
||||
<div class="item-header-content">
|
||||
<div class="item-header">
|
||||
<p class="card-title">{{plant.PlantName}}</p>
|
||||
</div>
|
||||
<div class="item-Measurements-IconsAndNames">
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/air_humidity_icon.png" alt="not available">
|
||||
Air Humidity: <br>
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/air_temperature_icon.png" alt="not available">
|
||||
Air Temperature: <br>
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/soil_moisture_icon.png" alt="not available">
|
||||
Soil Moisture: <br>
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/brightness_icon.png" alt="not available">
|
||||
Brightness:<br>
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/Timestamp_icon.png" alt="not available">
|
||||
Timestamp:
|
||||
</div>
|
||||
<div class="item-Measurements-Values">
|
||||
<p>{{plant.AirHumidity}} %</p>
|
||||
<p>{{plant.AirTemperature}} °C</p>
|
||||
<p>{{plant.SoilMoisture}} %</p>
|
||||
<p>{{plant.Brightness}} Lux</p>
|
||||
<p>{{getFormattedLastRequestedTimestamp() | async}}</p>
|
||||
</div>
|
||||
</div>
|
||||
<div class="item-IMG">
|
||||
<img class="IMG-Plants" src="assets/images/gruppe1.jpg" alt="Plant is not available">
|
||||
<img class="IMG-Plants" src="assets/images/gruppe2.jpg" alt="Plant is not available">
|
||||
</div>
|
||||
<div class="item-delete">
|
||||
<button class="item-Configure" mat-icon-button color="primary" (click)="onConfigurePlant()">
|
||||
<mat-icon>create</mat-icon>
|
||||
</button>
|
||||
<button mat-icon-button color="warn" (click)="onDeletePlant()">
|
||||
<mat-icon class="delete-icon">delete</mat-icon>
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
@ -0,0 +1,23 @@
|
||||
// import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { PlantCardComponent } from './plant-card.component';
|
||||
|
||||
// describe('PlantCardComponent', () => {
|
||||
// let component: PlantCardComponent;
|
||||
// let fixture: ComponentFixture<PlantCardComponent>;
|
||||
|
||||
// beforeEach(async () => {
|
||||
// await TestBed.configureTestingModule({
|
||||
// declarations: [PlantCardComponent]
|
||||
// })
|
||||
// .compileComponents();
|
||||
|
||||
// fixture = TestBed.createComponent(PlantCardComponent);
|
||||
// component = fixture.componentInstance;
|
||||
// fixture.detectChanges();
|
||||
// });
|
||||
|
||||
// it('should create', () => {
|
||||
// expect(component).toBeTruthy();
|
||||
// });
|
||||
// });
|
@ -0,0 +1,62 @@
|
||||
import { Component, Input } from '@angular/core';
|
||||
import { MatDialog } from '@angular/material/dialog';
|
||||
import { Observable, of } from 'rxjs';
|
||||
import { MqttRequestService } from 'src/app/Service/Mqtt/mqtt-request.service';
|
||||
import { ConfigurePlantDialogComponent } from 'src/app/dialog/configure-plant-dialog/configure-plant-dialog.component';
|
||||
import { DeleteConfirmDialogComponent } from 'src/app/dialog/delete-confirm-dialog/delete-confirm-dialog.component';
|
||||
import { Plant } from 'src/app/models/plant.model';
|
||||
|
||||
@Component({
|
||||
selector: 'app-plant-card',
|
||||
templateUrl: './plant-card.component.html',
|
||||
styleUrls: ['./plant-card.component.css']
|
||||
})
|
||||
export class PlantCardComponent {
|
||||
@Input() plant: Plant;
|
||||
currentDate: Date;
|
||||
|
||||
constructor(public dialog: MatDialog, private mqttRequestService: MqttRequestService) {
|
||||
}
|
||||
|
||||
getFormattedLastRequestedTimestamp(): Observable<string> {
|
||||
return of(this.calculateTimestamp(this.plant.Timestamp))
|
||||
}
|
||||
|
||||
calculateTimestamp(timestamp: string): string {
|
||||
const datePlant = new Date(timestamp);
|
||||
this.currentDate = new Date();
|
||||
let seconds = ((this.currentDate.getTime() - datePlant.getTime()) / (1000));
|
||||
|
||||
const minutes = Math.floor(seconds / 60);
|
||||
const hours = Math.floor(minutes / 60);
|
||||
const days = Math.floor(hours / 24);
|
||||
|
||||
if (days >= 1) {
|
||||
return `${days} day${days > 1 ? 's' : ''} ago`;
|
||||
} else if (hours >= 1) {
|
||||
return `${hours} hour${hours > 1 ? 's' : ''} ago`;
|
||||
} else if (minutes >= 1) {
|
||||
return `${minutes} minute${minutes > 1 ? 's' : ''} ago`;
|
||||
} else {
|
||||
seconds = Math.floor(seconds);
|
||||
return `${seconds} second${seconds !== 1 ? 's' : ''} ago`;
|
||||
}
|
||||
}
|
||||
|
||||
onDeletePlant() {
|
||||
this.dialog.open(DeleteConfirmDialogComponent, {
|
||||
data: this.plant,
|
||||
width: '400px',
|
||||
});
|
||||
}
|
||||
|
||||
onConfigurePlant() {
|
||||
this.dialog.open(ConfigurePlantDialogComponent, {
|
||||
data: {
|
||||
plant: this.plant
|
||||
},
|
||||
width: '400px',
|
||||
});
|
||||
}
|
||||
|
||||
}
|
39
software/Frontend/src/app/Plants/plants.component.css
Normal file
@ -0,0 +1,39 @@
|
||||
.container {
|
||||
display: grid;
|
||||
width: 100%;
|
||||
grid-template-columns: 50% 50%;
|
||||
grid-template-rows: 20% 80%;
|
||||
margin-top: 3.125rem;
|
||||
}
|
||||
|
||||
.buttonSearch {
|
||||
background-color: white;
|
||||
border: none;
|
||||
}
|
||||
|
||||
.item-Search {
|
||||
grid-row: 1 / 2;
|
||||
grid-column: 1/2;
|
||||
justify-self: end;
|
||||
align-self: center;
|
||||
margin-right: 8rem;
|
||||
}
|
||||
|
||||
.item-NewPlant {
|
||||
grid-row: 1 / 2;
|
||||
grid-column: 2/3;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
margin-left: 18rem;
|
||||
}
|
||||
|
||||
.item-Cards {
|
||||
grid-row: 2 / -1;
|
||||
grid-column: 1/3;
|
||||
justify-self: center;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
mat-spinner {
|
||||
margin: 7rem auto;
|
||||
}
|
32
software/Frontend/src/app/Plants/plants.component.html
Normal file
@ -0,0 +1,32 @@
|
||||
<div class="container" *ngIf="(plants$ |async) !== undefined; else errorLoadingData">
|
||||
<div class="item-Search">
|
||||
<form class="example-form">
|
||||
<mat-form-field class=" example-full-width" appearance="fill">
|
||||
<input type="text" placeholder="Plant Name" aria-label="Plant Name" matInput [formControl]="myControl"
|
||||
[matAutocomplete]="auto">
|
||||
<mat-autocomplete autoActiveFirstOption #auto="matAutocomplete"
|
||||
(optionSelected)='onOptionSelect($event.option.value)'>
|
||||
<mat-option *ngFor="let option of filteredOptions | async" [value]="option">
|
||||
{{option}}
|
||||
</mat-option>
|
||||
</mat-autocomplete>
|
||||
</mat-form-field>
|
||||
<button class="buttonSearch"> <mat-icon matSuffix>search</mat-icon></button>
|
||||
</form>
|
||||
</div>
|
||||
<div class="example-button-row item-NewPlant">
|
||||
<button class="buttonNew" mat-raised-button color="primary" (click)="onNewPlant()">New Plant</button>
|
||||
</div>
|
||||
<div class="item-Cards">
|
||||
<div *ngIf="selectedPlant === 'undefined'; else elseBlock">
|
||||
<app-plant-card *ngFor="let plant of (plants$ | async)" [plant]="plant"></app-plant-card>
|
||||
</div>
|
||||
<ng-template #elseBlock>
|
||||
<app-plant-card [plant]="onSearchPlant(selectedPlant) | async"></app-plant-card>
|
||||
<button mat-raised-button color="primary" (click)="onOptionDefault()">Show all
|
||||
Plants</button>
|
||||
</ng-template>
|
||||
</div>
|
||||
</div>
|
||||
<ng-template #errorLoadingData><mat-spinner></mat-spinner>
|
||||
</ng-template>
|
23
software/Frontend/src/app/Plants/plants.component.spec.ts
Normal file
@ -0,0 +1,23 @@
|
||||
// import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { PlantsComponent } from './plants.component';
|
||||
|
||||
// describe('PlantsComponent', () => {
|
||||
// let component: PlantsComponent;
|
||||
// let fixture: ComponentFixture<PlantsComponent>;
|
||||
|
||||
// beforeEach(async () => {
|
||||
// await TestBed.configureTestingModule({
|
||||
// declarations: [ PlantsComponent ]
|
||||
// })
|
||||
// .compileComponents();
|
||||
|
||||
// fixture = TestBed.createComponent(PlantsComponent);
|
||||
// component = fixture.componentInstance;
|
||||
// fixture.detectChanges();
|
||||
// });
|
||||
|
||||
// it('should create', () => {
|
||||
// expect(component).toBeTruthy();
|
||||
// });
|
||||
// });
|
56
software/Frontend/src/app/Plants/plants.component.ts
Normal file
@ -0,0 +1,56 @@
|
||||
import { Component } from '@angular/core';
|
||||
import { FormControl } from '@angular/forms';
|
||||
import { MatDialog } from '@angular/material/dialog';
|
||||
import { Observable, map, startWith } from 'rxjs';
|
||||
import { MqttRequestService } from '../Service/Mqtt/mqtt-request.service';
|
||||
import { StoreService } from '../Service/store.service';
|
||||
import { AddPlantDialogComponent } from '../dialog/add-plant-dialog/add-plant-dialog.component';
|
||||
import { Plant } from '../models/plant.model';
|
||||
|
||||
@Component({
|
||||
selector: 'app-plants',
|
||||
templateUrl: './plants.component.html',
|
||||
styleUrls: ['./plants.component.css']
|
||||
})
|
||||
export class PlantsComponent {
|
||||
myControl = new FormControl('');
|
||||
options$: Observable<string[]> = this.storeService.getAllPlantNames();
|
||||
selectedPlant: string = 'undefined';
|
||||
filteredOptions: Observable<string[]>;
|
||||
plants$: Observable<Plant[]> = this.storeService.currentPlants;
|
||||
|
||||
|
||||
constructor(public dialog: MatDialog, public storeService: StoreService, private mqttRequestService: MqttRequestService) {
|
||||
this.options$.subscribe(response => {
|
||||
this.filteredOptions = this.myControl.valueChanges.pipe(
|
||||
startWith(''),
|
||||
map(value => this._filter(value || '', response)),
|
||||
);
|
||||
|
||||
})
|
||||
}
|
||||
|
||||
onSearchPlant(plantName: string): Observable<Plant> {
|
||||
return this.storeService.getPlant(plantName);
|
||||
}
|
||||
|
||||
onNewPlant() {
|
||||
this.dialog.open(AddPlantDialogComponent, {
|
||||
width: '420px',
|
||||
});
|
||||
}
|
||||
|
||||
onOptionSelect(value: string) {
|
||||
this.selectedPlant = value;
|
||||
}
|
||||
|
||||
onOptionDefault() {
|
||||
this.selectedPlant = 'undefined';
|
||||
}
|
||||
|
||||
private _filter(value: string, options: string[]): string[] {
|
||||
const filterValue = value.toLowerCase();
|
||||
|
||||
return options?.filter(option => option.toLowerCase().includes(filterValue));
|
||||
}
|
||||
}
|
88
software/Frontend/src/app/Robot/robot.component.css
Normal file
@ -0,0 +1,88 @@
|
||||
img {
|
||||
border-radius: 0.5rem;
|
||||
width: 15.625rem;
|
||||
height: 12.5rem;
|
||||
margin: 0rem 0.3125rem;
|
||||
}
|
||||
|
||||
.container {
|
||||
text-align: center;
|
||||
display: grid;
|
||||
grid-template-columns: 40.625rem 12.5rem;
|
||||
grid-template-rows: 12.5rem 6.25rem 6.25rem;
|
||||
width: 53.125rem;
|
||||
min-width: 46.875rem;
|
||||
font-family: Cormorant;
|
||||
box-shadow: 0.3125rem 0.3125rem 0.3125rem silver;
|
||||
border-radius: 1.25rem;
|
||||
border-color: silver;
|
||||
border-width: 0.0625rem;
|
||||
border-style: solid;
|
||||
padding: 0.625rem;
|
||||
margin: auto;
|
||||
margin-top: 3rem;
|
||||
}
|
||||
|
||||
.robotImage {
|
||||
grid-row: 1 / -1;
|
||||
grid-column: 2 / 3;
|
||||
justify-self: end;
|
||||
align-self: start;
|
||||
}
|
||||
|
||||
.robotAkku {
|
||||
grid-row: 1 / 2;
|
||||
grid-column: 1/2;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
display: grid;
|
||||
grid-template-columns: 15.625rem 12.5rem 12.5rem;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.akkuMsg {
|
||||
grid-column: 1/2;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.akkuProgressbar {
|
||||
grid-column: 2/3;
|
||||
justify-self: center;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
.akkuValue {
|
||||
grid-column: 3/4;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
margin: 0;
|
||||
margin-left: 0.3125rem;
|
||||
}
|
||||
|
||||
.robotCoordinates {
|
||||
grid-row: 2 / 3;
|
||||
grid-column: 1/2;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
display: grid;
|
||||
grid-template-columns: 15.625rem 12.5rem;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.item-timestamp-name {
|
||||
grid-column: 1/2;
|
||||
justify-self: start;
|
||||
align-self: start;
|
||||
}
|
||||
|
||||
.item-timestamp-value {
|
||||
grid-column: 2/3;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
mat-spinner {
|
||||
margin: 7rem auto;
|
||||
}
|
18
software/Frontend/src/app/Robot/robot.component.html
Normal file
@ -0,0 +1,18 @@
|
||||
<div class="container"
|
||||
*ngIf="robotBatteryResponse$ !== undefined && (robotBatteryResponse$ |async) !== undefined; else errorLoadingData">
|
||||
<div class="robotImage">
|
||||
<img src="assets/images/gruppe1.jpg" alt="Plant is not available">
|
||||
</div>
|
||||
<div class="robotAkku">
|
||||
<h1 class="akkuMsg">Robot Akku:</h1>
|
||||
<mat-progress-bar class="akkuProgressbar" mode="determinate" [value]="batteryChargingStatus$ | async"
|
||||
[color]="(getLoadingBarColor() | async)"></mat-progress-bar>
|
||||
<h3 class="akkuValue">{{batteryChargingStatus$ | async}}%</h3>
|
||||
</div>
|
||||
<div class="robotCoordinates">
|
||||
<h1 class="item-timestamp-name">Timestamp:</h1>
|
||||
<h1 class="item-timestamp-value">{{getFormattedLastRequestedBatteryState()| async}}</h1>
|
||||
</div>
|
||||
</div>
|
||||
<ng-template #errorLoadingData><mat-spinner></mat-spinner>
|
||||
</ng-template>
|
23
software/Frontend/src/app/Robot/robot.component.spec.ts
Normal file
@ -0,0 +1,23 @@
|
||||
// import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { RobotComponent } from './robot.component';
|
||||
|
||||
// describe('RobotComponent', () => {
|
||||
// let component: RobotComponent;
|
||||
// let fixture: ComponentFixture<RobotComponent>;
|
||||
|
||||
// beforeEach(async () => {
|
||||
// await TestBed.configureTestingModule({
|
||||
// declarations: [ RobotComponent ]
|
||||
// })
|
||||
// .compileComponents();
|
||||
|
||||
// fixture = TestBed.createComponent(RobotComponent);
|
||||
// component = fixture.componentInstance;
|
||||
// fixture.detectChanges();
|
||||
// });
|
||||
|
||||
// it('should create', () => {
|
||||
// expect(component).toBeTruthy();
|
||||
// });
|
||||
// });
|
53
software/Frontend/src/app/Robot/robot.component.ts
Normal file
@ -0,0 +1,53 @@
|
||||
import { Component } from '@angular/core';
|
||||
import { ThemePalette } from '@angular/material/core';
|
||||
import { Observable, map } from 'rxjs';
|
||||
import { StoreService } from '../Service/store.service';
|
||||
import { RobotBattery } from '../models/robotBattery.model';
|
||||
|
||||
@Component({
|
||||
selector: 'app-robot',
|
||||
templateUrl: './robot.component.html',
|
||||
styleUrls: ['./robot.component.css']
|
||||
})
|
||||
export class RobotComponent {
|
||||
robotBatteryResponse$: Observable<RobotBattery> = this.storeService.currentRobotBattery;
|
||||
batteryChargingStatus$: Observable<number> = this.robotBatteryResponse$.pipe(map(response => response.Battery));
|
||||
requestTimestamp$: Observable<string> = this.robotBatteryResponse$.pipe(map(response => response.Timestamp));
|
||||
|
||||
constructor(private storeService: StoreService) {
|
||||
}
|
||||
|
||||
getLoadingBarColor(): Observable<ThemePalette> {
|
||||
return this.batteryChargingStatus$.pipe(
|
||||
map(chargingStatus => chargingStatus < 20 ? "warn" : "primary")
|
||||
)
|
||||
}
|
||||
|
||||
getFormattedLastRequestedBatteryState(): Observable<string> {
|
||||
return this.requestTimestamp$.pipe(
|
||||
map(timestamp => this.formatLastRequestedBatteryState(timestamp))
|
||||
)
|
||||
}
|
||||
|
||||
formatLastRequestedBatteryState(timestamp: string): string {
|
||||
const datePlant = new Date(timestamp);
|
||||
const currentDate = new Date();
|
||||
let seconds = ((currentDate.getTime() - datePlant.getTime()) / (1000));
|
||||
|
||||
const minutes = Math.floor(seconds / 60);
|
||||
const hours = Math.floor(minutes / 60);
|
||||
const days = Math.floor(hours / 24);
|
||||
|
||||
if (days >= 1) {
|
||||
return `${days} day${days > 1 ? 's' : ''} ago`;
|
||||
} else if (hours >= 1) {
|
||||
return `${hours} hour${hours > 1 ? 's' : ''} ago`;
|
||||
} else if (minutes >= 1) {
|
||||
return `${minutes} minute${minutes > 1 ? 's' : ''} ago`;
|
||||
} else {
|
||||
seconds = Math.floor(seconds);
|
||||
return `${seconds} second${seconds !== 1 ? 's' : ''} ago`;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
// import { TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { MqttRequestService } from './mqtt-request.service';
|
||||
|
||||
// describe('MqttRequestService', () => {
|
||||
// let service: MqttRequestService;
|
||||
|
||||
// beforeEach(() => {
|
||||
// TestBed.configureTestingModule({});
|
||||
// service = TestBed.inject(MqttRequestService);
|
||||
// });
|
||||
|
||||
// it('should be created', () => {
|
||||
// expect(service).toBeTruthy();
|
||||
// });
|
||||
// });
|
222
software/Frontend/src/app/Service/Mqtt/mqtt-request.service.ts
Normal file
@ -0,0 +1,222 @@
|
||||
import { Injectable } from '@angular/core';
|
||||
import { MatDialog } from '@angular/material/dialog';
|
||||
import { BehaviorSubject, Subscription } from 'rxjs';
|
||||
import { ErrorDialogComponent } from 'src/app/dialog/error-dialog/error-dialog.component';
|
||||
import { Plant } from 'src/app/models/plant.model';
|
||||
import { PlantCount } from 'src/app/models/plantCount.model';
|
||||
import { Position } from 'src/app/models/position.model';
|
||||
import { RobotBattery } from 'src/app/models/robotBattery.model';
|
||||
import { StoreService } from '../store.service';
|
||||
import { MqttSetDataService } from './mqtt-set-data.service';
|
||||
import { MqttService } from './mqtt.service';
|
||||
|
||||
@Injectable({
|
||||
providedIn: 'root'
|
||||
})
|
||||
export class MqttRequestService {
|
||||
private sourceMessage = new BehaviorSubject('');
|
||||
responseMessage = this.sourceMessage.asObservable();
|
||||
|
||||
timerSubscription: Subscription | undefined;
|
||||
|
||||
constructor(private mqttService: MqttService,
|
||||
private mqttSetDataService: MqttSetDataService,
|
||||
private storeService: StoreService, public dialog: MatDialog) {
|
||||
//Subscribe to the Topics
|
||||
this.mqttService.subscribeToTopic("BACKEND/DATA/SENSORDATA_ALL").subscribe(data => {
|
||||
if (typeof data !== "object") {
|
||||
const payload = JSON.parse(data) as Plant[];
|
||||
this.mqttSetDataService.setDataAllPlants(payload);
|
||||
}
|
||||
}, err => {
|
||||
this.dialog.open(ErrorDialogComponent, {
|
||||
data: err,
|
||||
width: '400px',
|
||||
});
|
||||
console.error('Error:', err);
|
||||
});
|
||||
|
||||
this.mqttService.subscribeToTopic("BACKEND/DATA/ERROR").subscribe(data => {
|
||||
if (typeof data !== "object") {
|
||||
const payload = JSON.parse(data) as string;
|
||||
this.dialog.open(ErrorDialogComponent, {
|
||||
data: payload,
|
||||
width: '400px',
|
||||
});
|
||||
}
|
||||
}, err => {
|
||||
this.dialog.open(ErrorDialogComponent, {
|
||||
data: err,
|
||||
width: '400px',
|
||||
});
|
||||
console.error('Error:', err);
|
||||
});
|
||||
|
||||
this.mqttService.subscribeToTopic("BACKEND/DATA/SENSORDATA").subscribe(data => {
|
||||
if (typeof data !== "object") {
|
||||
const payload = JSON.parse(data) as Plant;
|
||||
this.mqttSetDataService.setDataPlant(payload);
|
||||
}
|
||||
}, err => {
|
||||
this.dialog.open(ErrorDialogComponent, {
|
||||
data: err,
|
||||
width: '400px',
|
||||
});
|
||||
console.error('Error:', err);
|
||||
});
|
||||
|
||||
this.mqttService.subscribeToTopic("BACKEND/DATA/POSITION").subscribe(data => {
|
||||
if (typeof data !== "object") {
|
||||
const payload = JSON.parse(data) as Position;
|
||||
this.mqttSetDataService.setRobotPosition(payload);
|
||||
}
|
||||
}, err => {
|
||||
this.dialog.open(ErrorDialogComponent, {
|
||||
data: err,
|
||||
width: '400px',
|
||||
});
|
||||
console.error('Error:', err);
|
||||
});
|
||||
|
||||
this.mqttService.subscribeToTopic("BACKEND/DATA/BATTERY").subscribe(data => {
|
||||
if (typeof data !== "object") {
|
||||
const payload = JSON.parse(data) as RobotBattery;
|
||||
this.mqttSetDataService.setRobotBattery(payload);
|
||||
}
|
||||
}, err => {
|
||||
this.dialog.open(ErrorDialogComponent, {
|
||||
data: err,
|
||||
width: '400px',
|
||||
});
|
||||
console.error('Error:', err);
|
||||
});
|
||||
|
||||
this.mqttService.subscribeToTopic("BACKEND/DATA/ROBOTREADY").subscribe(data => {
|
||||
if (typeof data !== "object") {
|
||||
const payload = JSON.parse(data) as boolean;
|
||||
this.mqttSetDataService.setRobotReady(payload);
|
||||
}
|
||||
}, err => {
|
||||
this.dialog.open(ErrorDialogComponent, {
|
||||
data: err,
|
||||
width: '400px',
|
||||
});
|
||||
console.error('Error:', err);
|
||||
});
|
||||
|
||||
this.mqttService.subscribeToTopic("BACKEND/DATA/PLANTCOUNT").subscribe(data => {
|
||||
if (typeof data !== "object") {
|
||||
const payload = JSON.parse(data) as PlantCount;
|
||||
this.mqttSetDataService.setPlantCount(payload);
|
||||
}
|
||||
}, err => {
|
||||
this.dialog.open(ErrorDialogComponent, {
|
||||
data: err,
|
||||
width: '400px',
|
||||
});
|
||||
console.error('Error:', err);
|
||||
});
|
||||
// this.mqttService.subscribeToTopic("BACKEND/DATA/POSITION").subscribe(data => {
|
||||
// if (typeof data !== "object") {
|
||||
// const payload = JSON.parse(data) as Plant[];
|
||||
// this.mqttSetDataService.setDataAllPlants(payload);
|
||||
// }
|
||||
// }, err => {
|
||||
// console.error('Error:', err);
|
||||
// });
|
||||
// this.mqttService.subscribeToTopic("BACKEND/DATA/PICTURE").subscribe(data => {
|
||||
// if (typeof data !== "object") {
|
||||
// const payload = JSON.parse(data) as Plant;
|
||||
// this.mqttSetDataService.setDataPlant(payload);
|
||||
// }
|
||||
// }, err => {
|
||||
// console.error('Error:', err);
|
||||
// });
|
||||
|
||||
//Request to get ALL DATA every 10sec!!
|
||||
// this.timerSubscription = timer(0, 10000).pipe(
|
||||
// map(() => {
|
||||
// this.mqttService.publishToTopic('BACKEND/ACTION/GETALLDATA');
|
||||
// })
|
||||
// ).subscribe();
|
||||
|
||||
//Publish to the Topic to recieve the first Data
|
||||
this.publishToPLANTCOUNT();
|
||||
this.publishToGETBATTERY();
|
||||
this.publishToROBOTREADY();
|
||||
this.mqttService.publishToTopic('BACKEND/ACTION/GETALLDATA', JSON.stringify(this.storeService.getAllPlantNames()));
|
||||
}
|
||||
|
||||
publishToGETALLDATA() {
|
||||
this.mqttService.publishToTopic('BACKEND/ACTION/GETALLDATA', JSON.stringify(this.storeService.getAllPlantNames()));
|
||||
}
|
||||
publishToDRIVE() {
|
||||
this.mqttService.publishToTopic('BACKEND/ACTION/DRIVE');
|
||||
}
|
||||
publishToGETBATTERY() {
|
||||
this.mqttService.publishToTopic('BACKEND/ACTION/GETBATTERY');
|
||||
}
|
||||
publishToROBOTREADY() {
|
||||
this.mqttService.publishToTopic('BACKEND/ACTION/ROBOTREADY');
|
||||
}
|
||||
publishToGETPOSITION() {
|
||||
this.mqttService.publishToTopic('BACKEND/ACTION/GETPOSITION');
|
||||
}
|
||||
|
||||
publishToNEWPLANT(plant: Plant) {
|
||||
this.mqttService.publishToTopic('BACKEND/ACTION/NEWPLANT', JSON.stringify(plant));
|
||||
}
|
||||
|
||||
publishToCONFIGUREPLANT(plant: Plant) {
|
||||
this.mqttService.publishToTopic('BACKEND/ACTION/CONFIGUREPLANT', JSON.stringify(plant));
|
||||
}
|
||||
|
||||
publishToDELETEPLANT(plantID: number) {
|
||||
this.mqttService.publishToTopic('BACKEND/ACTION/DELETEPLANT', JSON.stringify(plantID));
|
||||
}
|
||||
|
||||
publishToPLANTCOUNT() {
|
||||
this.mqttService.publishToTopic('BACKEND/ACTION/PLANTCOUNT');
|
||||
}
|
||||
|
||||
publishToINPUT(message: string) {
|
||||
message = message.replace(/\s/g, "");
|
||||
console.log(":" + message + ":");
|
||||
let msgFound: boolean = false;
|
||||
|
||||
if (message === "whereismyrobot") {
|
||||
msgFound = true;
|
||||
this.mqttService.publishToTopic("BACKEND/ACTION/GETPOSITION")
|
||||
this.sourceMessage.next("The data is being collected and you can view it under the 'Robot' tab");
|
||||
} else if (message === "howmuchbatterydoesmyrobothave") {
|
||||
msgFound = true;
|
||||
this.mqttService.publishToTopic('BACKEND/ACTION/GETBATTERY');
|
||||
this.sourceMessage.next("The data is being collected and you can view it under the 'Robot' tab");
|
||||
} else if (message === "howaremyplants") {
|
||||
msgFound = true;
|
||||
// this.mqttService.publishToTopic('BACKEND/ACTION/DRIVEALL');
|
||||
this.sourceMessage.next("Robot is driving to all plants");
|
||||
} else {
|
||||
this.storeService.getAllPlantNames().subscribe(result =>
|
||||
result.forEach((element: string) => {
|
||||
let msgPlantName = "howismyplant".concat(element.toLowerCase())
|
||||
if (message === msgPlantName) {
|
||||
msgFound = true;
|
||||
let PlantName = element;
|
||||
this.mqttService.publishToTopic('BACKEND/ACTION/DRIVE', JSON.stringify(PlantName));
|
||||
this.sourceMessage.next("Robot is driving to plant " + element);
|
||||
}
|
||||
})
|
||||
)
|
||||
}
|
||||
|
||||
if (!msgFound) {
|
||||
this.sourceMessage.next("Please try again");
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
onDestroy() {
|
||||
this.sourceMessage.next('');
|
||||
}
|
||||
}
|
@ -0,0 +1,16 @@
|
||||
// import { TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { MqttSetDataService } from './mqtt-set-data.service';
|
||||
|
||||
// describe('MqttSetDataService', () => {
|
||||
// let service: MqttSetDataService;
|
||||
|
||||
// beforeEach(() => {
|
||||
// TestBed.configureTestingModule({});
|
||||
// service = TestBed.inject(MqttSetDataService);
|
||||
// });
|
||||
|
||||
// it('should be created', () => {
|
||||
// expect(service).toBeTruthy();
|
||||
// });
|
||||
// });
|
@ -0,0 +1,36 @@
|
||||
import { Injectable } from '@angular/core';
|
||||
import { Plant } from 'src/app/models/plant.model';
|
||||
import { PlantCount } from 'src/app/models/plantCount.model';
|
||||
import { Position } from 'src/app/models/position.model';
|
||||
import { RobotBattery } from 'src/app/models/robotBattery.model';
|
||||
import { StoreService } from '../store.service';
|
||||
|
||||
@Injectable({
|
||||
providedIn: 'root'
|
||||
})
|
||||
export class MqttSetDataService {
|
||||
|
||||
constructor(private storeService: StoreService) { }
|
||||
|
||||
setDataAllPlants(data: Plant[]): void {
|
||||
this.storeService.setDataAllPlants(data);
|
||||
}
|
||||
|
||||
setDataPlant(data: Plant): void {
|
||||
this.storeService.setDataPlant(data);
|
||||
}
|
||||
|
||||
setRobotBattery(data: RobotBattery): void {
|
||||
this.storeService.setRobotState(data);
|
||||
}
|
||||
setRobotReady(data: boolean): void {
|
||||
this.storeService.setRobotReady(data);
|
||||
}
|
||||
setRobotPosition(data: Position): void {
|
||||
this.storeService.setRobotPosition(data);
|
||||
}
|
||||
setPlantCount(data: PlantCount): void {
|
||||
this.storeService.setPlantCount(data);
|
||||
}
|
||||
|
||||
}
|
16
software/Frontend/src/app/Service/Mqtt/mqtt.service.spec.ts
Normal file
@ -0,0 +1,16 @@
|
||||
// import { TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { MqttService } from './mqtt.service';
|
||||
|
||||
// describe('MqttService', () => {
|
||||
// let service: MqttService;
|
||||
|
||||
// beforeEach(() => {
|
||||
// TestBed.configureTestingModule({});
|
||||
// service = TestBed.inject(MqttService);
|
||||
// });
|
||||
|
||||
// it('should be created', () => {
|
||||
// expect(service).toBeTruthy();
|
||||
// });
|
||||
// });
|
37
software/Frontend/src/app/Service/Mqtt/mqtt.service.ts
Normal file
@ -0,0 +1,37 @@
|
||||
import { Injectable } from '@angular/core';
|
||||
import { connect, MqttClient } from 'mqtt/dist/mqtt';
|
||||
import { Observable } from 'rxjs';
|
||||
|
||||
@Injectable({
|
||||
providedIn: 'root'
|
||||
})
|
||||
export class MqttService {
|
||||
|
||||
private client: MqttClient;
|
||||
|
||||
constructor() {
|
||||
this.client = connect('wss://mqtt.eclipseprojects.io:443/mqtt'); //Je nachdem welchen Link der Broker hat
|
||||
// this.client = connect('mqtt://192.168.137.197:1883', { clientId: 'kemal' });
|
||||
}
|
||||
|
||||
public subscribeToTopic(topic: string): Observable<any> {
|
||||
return new Observable(observer => {
|
||||
this.client.subscribe(topic, (err, granted) => {
|
||||
if (err) {
|
||||
observer.error(err);
|
||||
} else {
|
||||
observer.next(granted);
|
||||
}
|
||||
});
|
||||
this.client.on('message', (t, m) => {
|
||||
if (t === topic) {
|
||||
observer.next(m.toString());
|
||||
}
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
public publishToTopic(topic: string, message?: any): void {
|
||||
this.client.publish(topic, message);
|
||||
}
|
||||
}
|
16
software/Frontend/src/app/Service/store.service.spec.ts
Normal file
@ -0,0 +1,16 @@
|
||||
import { TestBed } from '@angular/core/testing';
|
||||
|
||||
import { StoreService } from './store.service';
|
||||
|
||||
describe('StoreService', () => {
|
||||
let service: StoreService;
|
||||
|
||||
beforeEach(() => {
|
||||
TestBed.configureTestingModule({});
|
||||
service = TestBed.inject(StoreService);
|
||||
});
|
||||
|
||||
it('should be created', () => {
|
||||
expect(service).toBeTruthy();
|
||||
});
|
||||
});
|
96
software/Frontend/src/app/Service/store.service.ts
Normal file
@ -0,0 +1,96 @@
|
||||
import { Injectable } from '@angular/core';
|
||||
import { BehaviorSubject, Observable, map, of } from 'rxjs';
|
||||
import { Plant } from '../models/plant.model';
|
||||
import { PlantCount } from '../models/plantCount.model';
|
||||
import { Position } from '../models/position.model';
|
||||
import { RobotBattery } from '../models/robotBattery.model';
|
||||
|
||||
@Injectable({
|
||||
providedIn: 'root'
|
||||
})
|
||||
export class StoreService {
|
||||
positionRobot: Position;
|
||||
plantCount: PlantCount;
|
||||
|
||||
private plants: BehaviorSubject<Plant[]> = new BehaviorSubject<Plant[]>(undefined);
|
||||
currentPlants: Observable<Plant[]> = this.plants.asObservable();
|
||||
|
||||
private robotReady = new BehaviorSubject<boolean>(undefined);
|
||||
currentRobotReady = this.robotReady.asObservable();
|
||||
|
||||
private robotBattery = new BehaviorSubject<RobotBattery>(undefined);
|
||||
currentRobotBattery = this.robotBattery.asObservable();
|
||||
|
||||
constructor() {
|
||||
}
|
||||
|
||||
getPlant(name: string): Observable<Plant> {
|
||||
return this.currentPlants.pipe(map(response => {
|
||||
for (let plant of response) {
|
||||
if (plant.PlantName.toLocaleLowerCase() === name.toLocaleLowerCase()) {
|
||||
return plant;
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}))
|
||||
}
|
||||
|
||||
getAllPlantNames(): Observable<string[]> {
|
||||
return this.currentPlants.pipe(map(response => response?.map(response => response.PlantName)));
|
||||
}
|
||||
|
||||
getIDsWithoutPlant(plants: Plant[]): Observable<number[]> {
|
||||
let listOfID: number[] = [];
|
||||
let listOfIDWithoutPlant: number[] = [];
|
||||
|
||||
if (plants.length > 0) {
|
||||
//Write ID's of the Plants in a List
|
||||
plants.forEach(plant => {
|
||||
listOfID.push(plant.PlantID);
|
||||
});
|
||||
|
||||
//Sort the List of ID's
|
||||
listOfID.sort();
|
||||
|
||||
//Write the ID's without Plants in a List
|
||||
for (let index = 0; index < this.plantCount.MaxCount; index++) { //Noch unsicher.... evtl. die länge von wo anders holen..
|
||||
if (!listOfID.includes(index + 1)) {
|
||||
listOfIDWithoutPlant.push(index + 1);
|
||||
}
|
||||
}
|
||||
return of(listOfIDWithoutPlant);
|
||||
} else {
|
||||
for (let index = 0; index < this.plantCount.MaxCount; index++) {
|
||||
listOfIDWithoutPlant.push(index + 1);
|
||||
}
|
||||
return of(listOfIDWithoutPlant);
|
||||
}
|
||||
}
|
||||
|
||||
//Set data
|
||||
setDataAllPlants(data: Plant[]): void {
|
||||
this.plants.next(data);
|
||||
}
|
||||
|
||||
setDataPlant(data: Plant): void {
|
||||
this.plants.subscribe(response => {
|
||||
response.forEach(plant => {
|
||||
if (plant.PlantID === data.PlantID) {
|
||||
plant = data;
|
||||
}
|
||||
})
|
||||
})
|
||||
}
|
||||
setRobotState(data: RobotBattery): void {
|
||||
this.robotBattery.next(data);
|
||||
}
|
||||
setRobotReady(data: boolean): void {
|
||||
this.robotReady.next(data);
|
||||
}
|
||||
setRobotPosition(data: Position): void {
|
||||
this.positionRobot = data;
|
||||
}
|
||||
setPlantCount(data: PlantCount): void {
|
||||
this.plantCount = data;
|
||||
}
|
||||
}
|
10
software/Frontend/src/app/app-routing.module.ts
Normal file
@ -0,0 +1,10 @@
|
||||
import { NgModule } from '@angular/core';
|
||||
import { RouterModule, Routes } from '@angular/router';
|
||||
|
||||
const routes: Routes = [];
|
||||
|
||||
@NgModule({
|
||||
imports: [RouterModule.forRoot(routes)],
|
||||
exports: [RouterModule]
|
||||
})
|
||||
export class AppRoutingModule { }
|
44
software/Frontend/src/app/app.component.css
Normal file
@ -0,0 +1,44 @@
|
||||
.container {
|
||||
margin: 50px;
|
||||
}
|
||||
|
||||
.icon-menu {
|
||||
transform: scale(2);
|
||||
}
|
||||
|
||||
span {
|
||||
font-size: 1.5rem;
|
||||
font-family: Cormorant;
|
||||
}
|
||||
|
||||
header {
|
||||
border: 0.125rem;
|
||||
border-style: solid;
|
||||
border-radius: 1rem;
|
||||
border-color: #89bf54;
|
||||
box-shadow: 0.2rem 0.2rem 0.2rem silver;
|
||||
padding: 4px;
|
||||
display: grid;
|
||||
grid-template-columns: 20% 60% 20%;
|
||||
font-family: Cormorant;
|
||||
|
||||
.menu-item {
|
||||
grid-column: 1/2;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
.header-item {
|
||||
grid-column: 2/3;
|
||||
justify-self: center;
|
||||
align-self: end;
|
||||
font-size: 3rem;
|
||||
}
|
||||
|
||||
.img-item {
|
||||
grid-column: 3/4;
|
||||
justify-self: end;
|
||||
width: 15rem;
|
||||
height: 3rem;
|
||||
}
|
||||
}
|
36
software/Frontend/src/app/app.component.html
Normal file
@ -0,0 +1,36 @@
|
||||
<html class="container">
|
||||
|
||||
<header>
|
||||
<div class="menu-item">
|
||||
<button mat-icon-button [matMenuTriggerFor]="menu" aria-label="Example icon-button with a menu">
|
||||
<mat-icon class="icon-menu">menu</mat-icon>
|
||||
</button>
|
||||
<mat-menu #menu="matMenu">
|
||||
<button mat-menu-item routerLink="/home" (click)="onMenuSelect('Home')">
|
||||
<mat-icon>micro</mat-icon>
|
||||
<span>Home</span>
|
||||
</button>
|
||||
<button mat-menu-item routerLink="/map" (click)="onMenuSelect('Map')">
|
||||
<mat-icon>public</mat-icon>
|
||||
<span>Map</span>
|
||||
</button>
|
||||
<button mat-menu-item routerLink="plants" (click)="onMenuSelect('Plants')">
|
||||
<mat-icon>grass</mat-icon>
|
||||
<span>Plants</span>
|
||||
</button>
|
||||
<button mat-menu-item routerLink="robot" (click)="onMenuSelect('Robot')">
|
||||
<mat-icon>android</mat-icon>
|
||||
<span>Robot</span>
|
||||
</button>
|
||||
</mat-menu>
|
||||
</div>
|
||||
<p class="header-item">{{header}}</p>
|
||||
<img class="img-item" src="assets/images/logo.png" alt="not available">
|
||||
</header>
|
||||
|
||||
|
||||
<body>
|
||||
<router-outlet></router-outlet>
|
||||
</body>
|
||||
|
||||
</html>
|
59
software/Frontend/src/app/app.component.spec.ts
Normal file
@ -0,0 +1,59 @@
|
||||
import { Location } from '@angular/common';
|
||||
import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
import { MatMenuModule } from '@angular/material/menu';
|
||||
import { MqttRequestService } from './Service/Mqtt/mqtt-request.service';
|
||||
import { AppComponent } from './app.component';
|
||||
|
||||
describe('AppComponent', () => {
|
||||
let component: AppComponent;
|
||||
let fixture: ComponentFixture<AppComponent>;
|
||||
let mqttRequestServiceSpy: jasmine.SpyObj<MqttRequestService>;
|
||||
let locationStub: Partial<Location>;
|
||||
|
||||
beforeEach(async () => {
|
||||
const mqttRequestServiceMock = jasmine.createSpyObj('MqttRequestService', ['someMethod']);
|
||||
|
||||
locationStub = {
|
||||
path: () => '/home'
|
||||
};
|
||||
|
||||
await TestBed.configureTestingModule({
|
||||
imports: [MatMenuModule],
|
||||
declarations: [AppComponent],
|
||||
providers: [
|
||||
{ provide: MqttRequestService, useValue: mqttRequestServiceMock },
|
||||
{ provide: Location, useValue: locationStub }
|
||||
]
|
||||
}).compileComponents();
|
||||
});
|
||||
|
||||
beforeEach(() => {
|
||||
fixture = TestBed.createComponent(AppComponent);
|
||||
component = fixture.componentInstance;
|
||||
|
||||
mqttRequestServiceSpy = TestBed.inject(MqttRequestService) as jasmine.SpyObj<MqttRequestService>;
|
||||
|
||||
fixture.detectChanges();
|
||||
});
|
||||
|
||||
it('should create', () => {
|
||||
expect(component).toBeTruthy();
|
||||
});
|
||||
|
||||
it('should set header to "Home" if the route is empty', () => {
|
||||
expect(component.header).toBe('Home');
|
||||
});
|
||||
|
||||
it('should set header to the capitalized route value', () => {
|
||||
locationStub.path = () => '/about';
|
||||
const componentNew = new AppComponent(mqttRequestServiceSpy, locationStub as Location);
|
||||
|
||||
expect(componentNew.header).toBe('About');
|
||||
});
|
||||
|
||||
it('should update header when onMenuSelect is called', () => {
|
||||
component.onMenuSelect('Map');
|
||||
|
||||
expect(component.header).toBe('Map');
|
||||
});
|
||||
});
|
27
software/Frontend/src/app/app.component.ts
Normal file
@ -0,0 +1,27 @@
|
||||
import { Location } from '@angular/common';
|
||||
import { Component } from '@angular/core';
|
||||
|
||||
import { MqttRequestService } from './Service/Mqtt/mqtt-request.service';
|
||||
|
||||
@Component({
|
||||
selector: 'app-root',
|
||||
templateUrl: './app.component.html',
|
||||
styleUrls: ['./app.component.css'],
|
||||
})
|
||||
export class AppComponent {
|
||||
header: string;
|
||||
constructor(private mqttRequestService: MqttRequestService, private location: Location) {
|
||||
const actualRoute = this.location.path().slice(1);
|
||||
const lastParam = actualRoute.charAt(0).toUpperCase() + actualRoute.slice(1);
|
||||
if (lastParam === "") {
|
||||
this.onMenuSelect('Home');
|
||||
} else {
|
||||
this.onMenuSelect(lastParam);
|
||||
}
|
||||
}
|
||||
|
||||
onMenuSelect(header: string) {
|
||||
this.header = header;
|
||||
}
|
||||
}
|
||||
|
87
software/Frontend/src/app/app.module.ts
Normal file
@ -0,0 +1,87 @@
|
||||
import { NgModule } from '@angular/core';
|
||||
import { BrowserModule } from '@angular/platform-browser';
|
||||
|
||||
import { ScrollingModule } from '@angular/cdk/scrolling';
|
||||
import { HttpClientModule } from '@angular/common/http';
|
||||
import { FormsModule, ReactiveFormsModule } from "@angular/forms";
|
||||
import { MatAutocompleteModule } from '@angular/material/autocomplete';
|
||||
import { MatBadgeModule } from '@angular/material/badge';
|
||||
import { MatButtonModule } from '@angular/material/button';
|
||||
import { MatCardModule } from '@angular/material/card';
|
||||
import { MatDialogModule } from '@angular/material/dialog';
|
||||
import { MatFormFieldModule } from '@angular/material/form-field';
|
||||
import { MatIconModule } from '@angular/material/icon';
|
||||
import { MatInputModule } from '@angular/material/input';
|
||||
import { MatMenuModule } from '@angular/material/menu';
|
||||
import { MatProgressBarModule } from '@angular/material/progress-bar';
|
||||
import { MatProgressSpinnerModule } from '@angular/material/progress-spinner';
|
||||
import { MatSelectModule } from '@angular/material/select';
|
||||
import { BrowserAnimationsModule } from '@angular/platform-browser/animations';
|
||||
import { RouterModule, Routes } from '@angular/router';
|
||||
import { HomeComponent } from './Home/home.component';
|
||||
import { InputComponent } from './Home/input/input.component';
|
||||
import { OutputComponent } from './Home/output/output.component';
|
||||
import { MapComponent } from './Map/map.component';
|
||||
import { PlantCardComponent } from './Plants/plant-card/plant-card.component';
|
||||
import { PlantsComponent } from './Plants/plants.component';
|
||||
import { RobotComponent } from './Robot/robot.component';
|
||||
import { AppRoutingModule } from './app-routing.module';
|
||||
import { AppComponent } from './app.component';
|
||||
import { AddPlantDialogComponent } from './dialog/add-plant-dialog/add-plant-dialog.component';
|
||||
import { ConfigurePlantDialogComponent } from './dialog/configure-plant-dialog/configure-plant-dialog.component';
|
||||
import { ErrorDialogComponent } from './dialog/error-dialog/error-dialog.component';
|
||||
import { MapPlantDialogComponent } from './dialog/map-plant-dialog/map-plant-dialog.component';
|
||||
import { DeleteConfirmDialogComponent } from './dialog/delete-confirm-dialog/delete-confirm-dialog.component';
|
||||
|
||||
|
||||
const appRoutes: Routes = [
|
||||
{ path: '', component: HomeComponent },
|
||||
{ path: 'home', component: HomeComponent },
|
||||
{ path: 'map', component: MapComponent },
|
||||
{ path: 'plants', component: PlantsComponent },
|
||||
{ path: 'robot', component: RobotComponent }
|
||||
];
|
||||
|
||||
@NgModule({
|
||||
declarations: [
|
||||
AppComponent,
|
||||
HomeComponent,
|
||||
MapComponent,
|
||||
InputComponent,
|
||||
OutputComponent,
|
||||
PlantsComponent,
|
||||
PlantCardComponent,
|
||||
RobotComponent,
|
||||
ConfigurePlantDialogComponent,
|
||||
AddPlantDialogComponent,
|
||||
MapPlantDialogComponent,
|
||||
ErrorDialogComponent,
|
||||
DeleteConfirmDialogComponent
|
||||
],
|
||||
imports: [
|
||||
BrowserModule,
|
||||
AppRoutingModule,
|
||||
MatMenuModule,
|
||||
MatIconModule,
|
||||
MatButtonModule,
|
||||
HttpClientModule,
|
||||
BrowserAnimationsModule,
|
||||
MatBadgeModule,
|
||||
RouterModule.forRoot(appRoutes),
|
||||
MatFormFieldModule,
|
||||
FormsModule,
|
||||
MatInputModule,
|
||||
MatCardModule,
|
||||
ScrollingModule,
|
||||
MatProgressBarModule,
|
||||
MatAutocompleteModule,
|
||||
ReactiveFormsModule,
|
||||
MatDialogModule,
|
||||
MatSelectModule,
|
||||
MatProgressSpinnerModule
|
||||
],
|
||||
providers: [],
|
||||
bootstrap: [AppComponent],
|
||||
exports: [ConfigurePlantDialogComponent, AddPlantDialogComponent, MapPlantDialogComponent, ErrorDialogComponent]
|
||||
})
|
||||
export class AppModule { }
|
@ -0,0 +1,78 @@
|
||||
.item-header-content {
|
||||
display: grid;
|
||||
grid-template-columns: 11.25rem 6.25rem;
|
||||
grid-template-rows: 20% 80%;
|
||||
justify-content: start;
|
||||
align-content: start;
|
||||
}
|
||||
|
||||
.item-header {
|
||||
grid-column: 1/-1;
|
||||
grid-row: 1/2;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
font-size: 1.5625rem;
|
||||
margin-bottom: 1rem;
|
||||
color: blue;
|
||||
display: grid;
|
||||
grid-template-columns: 11.25rem 6.25rem;
|
||||
}
|
||||
|
||||
.item-header-text {
|
||||
grid-column: 1/2;
|
||||
justify-self: center;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
.item-header-button {
|
||||
grid-column: 2/3;
|
||||
justify-self: end;
|
||||
background-color: white;
|
||||
transform: scale(0.8);
|
||||
}
|
||||
|
||||
.icon-language {
|
||||
color: black;
|
||||
transform: scale(1.6);
|
||||
}
|
||||
|
||||
.item-header-button:hover {
|
||||
background-color: silver;
|
||||
}
|
||||
|
||||
.item-Measurements-IconsAndNames {
|
||||
grid-column: 1/2;
|
||||
grid-row: 2/3;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
font-size: 1.125rem;
|
||||
}
|
||||
|
||||
.item-Measurements-Values {
|
||||
grid-column: 2/3;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
.item-IMG {
|
||||
justify-self: center;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
p {
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.IMG-Plants {
|
||||
border-radius: 0.5rem;
|
||||
width: 11.25rem;
|
||||
height: 9.375rem;
|
||||
margin: 0rem 0.3125rem;
|
||||
}
|
||||
|
||||
.icons-measurements {
|
||||
width: 1rem;
|
||||
height: 0.75rem;
|
||||
margin: 0rem 0.3125rem;
|
||||
margin-top: 0.4rem;
|
||||
}
|
@ -0,0 +1,43 @@
|
||||
<div class="item-header-content">
|
||||
<div class="item-header">
|
||||
<p class="item-header-text">Add Plant:</p>
|
||||
<button mat-fab class="item-header-button" (click)="onOpenMap()"><mat-icon
|
||||
class="icon-language">language</mat-icon></button>
|
||||
</div>
|
||||
<div class="item-Measurements-IconsAndNames">
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/id_icon.png" alt="not available">
|
||||
ID: <br><br><br>
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/name_icon.png" alt="not available">
|
||||
Name: <br>
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/air_humidity_icon.png" alt="not available">
|
||||
Air Humidity: <br>
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/air_temperature_icon.png" alt="not available">
|
||||
Air Temperature: <br>
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/soil_moisture_icon.png" alt="not available">
|
||||
Soil Moisture: <br>
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/brightness_icon.png" alt="not available">
|
||||
Brightness:<br>
|
||||
<!-- <img class="icons-measurements" src="assets/images/Icons-Card/Timestamp_icon.png" alt="not available">
|
||||
Timestamp: -->
|
||||
</div>
|
||||
<div class="item-Measurements-Values">
|
||||
<mat-form-field class="item-option" appearance="fill">
|
||||
<mat-label>Choose an ID</mat-label>
|
||||
<mat-select [(value)]="selected" (closed)="onOptionSelected(selected)">
|
||||
<mat-option *ngFor="let option of options$ | async" [value]="option">
|
||||
{{option}}
|
||||
</mat-option>
|
||||
</mat-select>
|
||||
</mat-form-field>
|
||||
<input type="text" id="inputName" maxlength="15">
|
||||
<input type="number" id="inputAirTemperature" max="100">
|
||||
<input type="number" id="inputAirHumidity" max="100">
|
||||
<input type="number" id="inputSoilMoisture" max="100">
|
||||
<input type="number" id="inputBrightness" max="100">
|
||||
</div>
|
||||
</div>
|
||||
<div mat-dialog-actions>
|
||||
<button mat-button mat-dialog-close>No</button>
|
||||
<button mat-button mat-dialog-close cdkFocusInitial (click)="onSaveDialog()"
|
||||
[disabled]="buttonSavedisabled">Save</button>
|
||||
</div>
|
@ -0,0 +1,172 @@
|
||||
// import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
// import { MAT_DIALOG_DATA, MatDialog, MatDialogRef } from '@angular/material/dialog';
|
||||
// import { of } from 'rxjs';
|
||||
// import { MqttRequestService } from 'src/app/Service/Mqtt/mqtt-request.service';
|
||||
// import { StoreService } from 'src/app/Service/store.service';
|
||||
// import { Plant } from 'src/app/models/plant.model';
|
||||
// import { AddPlantDialogComponent } from './add-plant-dialog.component';
|
||||
// const plantMock1: Plant = {
|
||||
// AirTemperature: 25,
|
||||
// AirHumidity: 60,
|
||||
// SoilMoisture: 50,
|
||||
// Brightness: 80,
|
||||
// PlantID: 1,
|
||||
// Timestamp: "2023-06-05T12:00:00",
|
||||
// MeasurementID: "abcd1234",
|
||||
// PlantName: "Plant1",
|
||||
// };
|
||||
// const plantMock2: Plant = {
|
||||
// AirTemperature: 22,
|
||||
// AirHumidity: 60,
|
||||
// SoilMoisture: 50,
|
||||
// Brightness: 80,
|
||||
// PlantID: 1,
|
||||
// Timestamp: "2023-06-05T12:00:00",
|
||||
// MeasurementID: "abcd1234",
|
||||
// PlantName: "Plant2",
|
||||
// };
|
||||
// const plantMock3: Plant = {
|
||||
// AirTemperature: 21,
|
||||
// AirHumidity: 60,
|
||||
// SoilMoisture: 50,
|
||||
// Brightness: 80,
|
||||
// PlantID: 1,
|
||||
// Timestamp: "2023-06-05T12:00:00",
|
||||
// MeasurementID: "abcd1234",
|
||||
// PlantName: "Plant3",
|
||||
// };
|
||||
// describe('AddPlantDialogComponent', () => {
|
||||
// let component: AddPlantDialogComponent;
|
||||
// let fixture: ComponentFixture<AddPlantDialogComponent>;
|
||||
// let dialogRefSpyObj: jasmine.SpyObj<MatDialogRef<AddPlantDialogComponent>>;
|
||||
// let dialogSpy: jasmine.SpyObj<MatDialog>;
|
||||
// let mqttRequestServiceSpy: jasmine.SpyObj<MqttRequestService>;
|
||||
// let storeServiceSpy: jasmine.SpyObj<StoreService>;
|
||||
|
||||
// beforeEach(async () => {
|
||||
// const dialogRefSpy = jasmine.createSpyObj('MatDialogRef', ['close']);
|
||||
// const dialogSpyObj = jasmine.createSpyObj('MatDialog', ['open']);
|
||||
// const mqttRequestServiceSpyObj = jasmine.createSpyObj('MqttRequestService', [
|
||||
// 'publishToPLANTCOUNT',
|
||||
// 'publishToNEWPLANT',
|
||||
// 'publishToGETALLDATA',
|
||||
// ]);
|
||||
// const storeServiceSpyObj = jasmine.createSpyObj('StoreService', [
|
||||
// 'currentPlants',
|
||||
// 'getIDsWithoutPlant',
|
||||
// 'getAllPlantNames',
|
||||
// ]);
|
||||
|
||||
// await TestBed.configureTestingModule({
|
||||
// declarations: [AddPlantDialogComponent],
|
||||
// providers: [
|
||||
// { provide: MatDialogRef, useValue: dialogRefSpy },
|
||||
// { provide: MAT_DIALOG_DATA, useValue: {} },
|
||||
// { provide: MatDialog, useValue: dialogSpyObj },
|
||||
// { provide: MqttRequestService, useValue: mqttRequestServiceSpyObj },
|
||||
// { provide: StoreService, useValue: storeServiceSpyObj },
|
||||
// ],
|
||||
// }).compileComponents();
|
||||
// });
|
||||
|
||||
// beforeEach(() => {
|
||||
// fixture = TestBed.createComponent(AddPlantDialogComponent);
|
||||
// component = fixture.componentInstance;
|
||||
// dialogRefSpyObj = TestBed.inject(MatDialogRef) as jasmine.SpyObj<MatDialogRef<AddPlantDialogComponent>>;
|
||||
// dialogSpy = TestBed.inject(MatDialog) as jasmine.SpyObj<MatDialog>;
|
||||
// mqttRequestServiceSpy = TestBed.inject(MqttRequestService) as jasmine.SpyObj<MqttRequestService>;
|
||||
// storeServiceSpy = TestBed.inject(StoreService) as jasmine.SpyObj<StoreService>;
|
||||
// });
|
||||
|
||||
// it('should initialize with buttonSavedisabled set to true', () => {
|
||||
// // Assert
|
||||
// expect(component.buttonSavedisabled).toBe(true);
|
||||
// });
|
||||
|
||||
// it('should call publishToPLANTCOUNT method and retrieve options', () => {
|
||||
// // Arrange
|
||||
// const currentPlantsMock = of([plantMock1, plantMock2, plantMock3]);
|
||||
// const getIDsWithoutPlantMock = jasmine.createSpy().and.returnValue(of([4, 5, 6]));
|
||||
// const getAllPlantNamesMock = jasmine.createSpy().and.returnValue(of(['Plant1', 'Plant2', 'Plant3']));
|
||||
|
||||
// // Act
|
||||
// storeServiceSpy.currentPlants = currentPlantsMock;
|
||||
// storeServiceSpy.getIDsWithoutPlant = getIDsWithoutPlantMock;
|
||||
// storeServiceSpy.getAllPlantNames = getAllPlantNamesMock;
|
||||
// fixture.detectChanges();
|
||||
|
||||
// // Assert
|
||||
// expect(mqttRequestServiceSpy.publishToPLANTCOUNT).toHaveBeenCalled();
|
||||
// expect(storeServiceSpy.getIDsWithoutPlant).toHaveBeenCalledWith([plantMock1, plantMock2, plantMock3]);
|
||||
// expect(storeServiceSpy.getAllPlantNames).toHaveBeenCalled();
|
||||
// expect(component.options$).toEqual(of([4, 5, 6]));
|
||||
// expect(component.plantNames).toEqual(['Plant1', 'Plant2', 'Plant3']);
|
||||
// });
|
||||
|
||||
// // it('should open the error dialog if plant input fails', () => {
|
||||
// // // Arrange
|
||||
// // const plantInputString = 'Plant Name is empty';
|
||||
|
||||
// // // Act
|
||||
// // component.plantInputTestFails = jasmine.createSpy().and.returnValue(plantInputString);
|
||||
// // component.onSaveDialog();
|
||||
|
||||
// // // Assert
|
||||
// // expect(dialogSpy.open).toHaveBeenCalledWith(ErrorDialogComponent, {
|
||||
// // data: 'Error: ' + plantInputString,
|
||||
// // width: '400px',
|
||||
// // });
|
||||
// // expect(mqttRequestServiceSpy.publishToNEWPLANT).not.toHaveBeenCalled();
|
||||
// // expect(mqttRequestServiceSpy.publishToGETALLDATA).not.toHaveBeenCalled();
|
||||
// // expect(window.location.reload).not.toHaveBeenCalled();
|
||||
// // });
|
||||
|
||||
// // it('should call the MQTT request service and reload the page if plant input is valid', () => {
|
||||
// // // Arrange
|
||||
// // const plantInputString = '';
|
||||
|
||||
// // // Act
|
||||
// // component.plantInputTestFails = jasmine.createSpy().and.returnValue(plantInputString);
|
||||
// // component.onSaveDialog();
|
||||
|
||||
// // // Assert
|
||||
// // expect(dialogSpy.open).not.toHaveBeenCalled();
|
||||
// // expect(mqttRequestServiceSpy.publishToNEWPLANT).toHaveBeenCalledWith(component.plant);
|
||||
// // expect(mqttRequestServiceSpy.publishToGETALLDATA).toHaveBeenCalled();
|
||||
// // expect(window.location.reload).toHaveBeenCalled();
|
||||
// // });
|
||||
|
||||
// // it('should open the map plant dialog', () => {
|
||||
// // // Act
|
||||
// // component.onOpenMap();
|
||||
|
||||
// // // Assert
|
||||
// // expect(dialogSpy.open).toHaveBeenCalledWith(MapPlantDialogComponent, {
|
||||
// // width: '400px',
|
||||
// // height: '400px',
|
||||
// // });
|
||||
// // });
|
||||
|
||||
// // it('should set buttonSavedisabled to false when option is selected within the range', () => {
|
||||
// // // Arrange
|
||||
// // component.buttonSavedisabled = true;
|
||||
|
||||
// // // Act
|
||||
// // component.onOptionSelected(5);
|
||||
|
||||
// // // Assert
|
||||
// // expect(component.buttonSavedisabled).toBeFalse();
|
||||
// // });
|
||||
|
||||
// // it('should keep buttonSavedisabled as true when option is selected outside the range', () => {
|
||||
// // // Arrange
|
||||
// // component.buttonSavedisabled = true;
|
||||
|
||||
// // // Act
|
||||
// // component.onOptionSelected(10);
|
||||
|
||||
// // // Assert
|
||||
// // expect(component.buttonSavedisabled).toBeTrue();
|
||||
// // });
|
||||
|
||||
// });
|
@ -0,0 +1,100 @@
|
||||
import { Component, Inject, OnInit } from '@angular/core';
|
||||
import { MAT_DIALOG_DATA, MatDialog, MatDialogRef } from '@angular/material/dialog';
|
||||
import { Observable } from 'rxjs';
|
||||
import { MqttRequestService } from 'src/app/Service/Mqtt/mqtt-request.service';
|
||||
import { StoreService } from 'src/app/Service/store.service';
|
||||
import { Plant } from 'src/app/models/plant.model';
|
||||
import { ErrorDialogComponent } from '../error-dialog/error-dialog.component';
|
||||
import { MapPlantDialogComponent } from '../map-plant-dialog/map-plant-dialog.component';
|
||||
|
||||
@Component({
|
||||
selector: 'app-add-plant-dialog',
|
||||
templateUrl: './add-plant-dialog.component.html',
|
||||
styleUrls: ['./add-plant-dialog.component.css']
|
||||
})
|
||||
export class AddPlantDialogComponent implements OnInit {
|
||||
selected: number;
|
||||
options$: Observable<number[]>;
|
||||
plantNames: string[];
|
||||
plant = {} as Plant;
|
||||
buttonSavedisabled: boolean;
|
||||
number: number[];
|
||||
|
||||
constructor(public dialogRef: MatDialogRef<AddPlantDialogComponent>,
|
||||
@Inject(MAT_DIALOG_DATA) public data: any,
|
||||
public dialog: MatDialog,
|
||||
private storeService: StoreService,
|
||||
private mqttRequestService: MqttRequestService) {
|
||||
}
|
||||
|
||||
ngOnInit(): void {
|
||||
this.buttonSavedisabled = true;
|
||||
this.mqttRequestService.publishToPLANTCOUNT();
|
||||
this.storeService.currentPlants.subscribe(response => {
|
||||
this.options$ = this.storeService.getIDsWithoutPlant(response);
|
||||
})
|
||||
this.storeService.getAllPlantNames().subscribe(result => {
|
||||
this.plantNames = result
|
||||
});
|
||||
}
|
||||
|
||||
onSaveDialog() {
|
||||
this.plant.PlantName = (<HTMLInputElement>document.getElementById("inputName")).value;
|
||||
this.plant.AirTemperature = +(<HTMLInputElement>document.getElementById("inputAirTemperature")).value;
|
||||
this.plant.AirHumidity = +(<HTMLInputElement>document.getElementById("inputAirHumidity")).value;
|
||||
this.plant.SoilMoisture = +(<HTMLInputElement>document.getElementById("inputSoilMoisture")).value;
|
||||
this.plant.Brightness = +(<HTMLInputElement>document.getElementById("inputBrightness")).value;
|
||||
this.plant.PlantID = this.selected;
|
||||
|
||||
const plantInputString: string = this.plantInputTestFails();
|
||||
|
||||
if (plantInputString) {
|
||||
this.dialog.open(ErrorDialogComponent, {
|
||||
data: "Error: " + plantInputString,
|
||||
width: '400px',
|
||||
});
|
||||
} else {
|
||||
this.mqttRequestService.publishToNEWPLANT(this.plant);
|
||||
this.mqttRequestService.publishToGETALLDATA();
|
||||
window.location.reload();
|
||||
}
|
||||
}
|
||||
|
||||
plantInputTestFails(): string {
|
||||
|
||||
if (this.plantNames.indexOf(this.plant.PlantName) > -1) {
|
||||
return "Plant Name already exists";
|
||||
}
|
||||
if (this.plant.PlantName === "") {
|
||||
return "Plant Name is empty";
|
||||
}
|
||||
if (this.plant.AirTemperature > 100) {
|
||||
return "Air Temperature is to high";
|
||||
}
|
||||
if (this.plant.AirHumidity > 100) {
|
||||
return "Air Humidity is to high";
|
||||
}
|
||||
if (this.plant.SoilMoisture > 100) {
|
||||
return "Soil Moisture is to high";
|
||||
}
|
||||
if (this.plant.Brightness > 100) {
|
||||
return "Brightness is to high";
|
||||
}
|
||||
return ""; //Empty string is a Boolean false!
|
||||
}
|
||||
|
||||
onOpenMap() {
|
||||
this.dialog.open(MapPlantDialogComponent, {
|
||||
width: '400px',
|
||||
height: '400px'
|
||||
});
|
||||
}
|
||||
|
||||
onOptionSelected(numb: number) {
|
||||
let num: number = numb;
|
||||
if (num < 7 && num > 0) { //TODOO anzahl der Pflanzen IDS variable machen statt 7 evtl. eine variable
|
||||
this.buttonSavedisabled = false;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,62 @@
|
||||
.item-header-content {
|
||||
display: grid;
|
||||
grid-template-columns: 11.25rem 6rem 3rem;
|
||||
grid-template-rows: 20% 80%;
|
||||
justify-content: start;
|
||||
align-content: start;
|
||||
}
|
||||
|
||||
.item-header {
|
||||
grid-column: 1/-1;
|
||||
grid-row: 1/2;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
font-size: 1.6rem;
|
||||
margin-bottom: 1rem;
|
||||
color: blue;
|
||||
}
|
||||
|
||||
.item-Measurements-IconsAndNames {
|
||||
grid-column: 1/2;
|
||||
grid-row: 2/3;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
.item-Measurements-Values {
|
||||
grid-column: 2/3;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
.item-Measurements-Units {
|
||||
grid-column: 3/4;
|
||||
justify-self: start;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
.item-IMG {
|
||||
justify-self: center;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
p {
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.IMG-Plants {
|
||||
border-radius: 0.5rem;
|
||||
width: 11.25rem;
|
||||
height: 9.375rem;
|
||||
margin: 0rem 0.3125rem;
|
||||
}
|
||||
|
||||
.icons-measurements {
|
||||
width: 1rem;
|
||||
height: 0.75rem;
|
||||
margin: 0rem 0.3125rem;
|
||||
}
|
||||
|
||||
input {
|
||||
width: 5rem;
|
||||
}
|
@ -0,0 +1,34 @@
|
||||
<div class="item-header-content">
|
||||
<div class="item-header">
|
||||
<p>Configure Plant: {{plant.PlantName}}</p>
|
||||
</div>
|
||||
<div class="item-Measurements-IconsAndNames">
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/air_humidity_icon.png" alt="not available">
|
||||
Air Humidity: <br>
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/air_temperature_icon.png" alt="not available">
|
||||
Air Temperature: <br>
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/soil_moisture_icon.png" alt="not available">
|
||||
Soil Moisture: <br>
|
||||
<img class="icons-measurements" src="assets/images/Icons-Card/brightness_icon.png" alt="not available">
|
||||
Brightness:<br>
|
||||
<!-- <img class="icons-measurements" src="assets/images/Icons-Card/Timestamp_icon.png" alt="not available">
|
||||
Timestamp: -->
|
||||
</div>
|
||||
<div class="item-Measurements-Values">
|
||||
<input type="number" [value]=plant.AirTemperature id="inputAirTemperature">
|
||||
<input type="number" [value]=plant.AirHumidity id="inputAirHumidity">
|
||||
<input type="number" [value]=plant.SoilMoisture id="inputSoilMoisture">
|
||||
<input type="number" [value]=plant.Brightness id="inputBrightness">
|
||||
<!-- <input type="datetime" [value]=plant.Timestamp id="inputTimestamp"> -->
|
||||
</div>
|
||||
<div class="item-Measurements-Units">
|
||||
<p>%</p>
|
||||
<p>°C</p>
|
||||
<p>%</p>
|
||||
<p>Lux</p>
|
||||
</div>
|
||||
</div>
|
||||
<div mat-dialog-actions>
|
||||
<button mat-button mat-dialog-close>No</button>
|
||||
<button mat-button mat-dialog-close cdkFocusInitial (click)="onSaveDialog()">Save</button>
|
||||
</div>
|
@ -0,0 +1,23 @@
|
||||
// import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { ConfigurePlantDialogComponent } from './configure-plant-dialog.component';
|
||||
|
||||
// describe('ConfigurePlantDialogComponent', () => {
|
||||
// let component: ConfigurePlantDialogComponent;
|
||||
// let fixture: ComponentFixture<ConfigurePlantDialogComponent>;
|
||||
|
||||
// beforeEach(async () => {
|
||||
// await TestBed.configureTestingModule({
|
||||
// declarations: [ ConfigurePlantDialogComponent ]
|
||||
// })
|
||||
// .compileComponents();
|
||||
|
||||
// fixture = TestBed.createComponent(ConfigurePlantDialogComponent);
|
||||
// component = fixture.componentInstance;
|
||||
// fixture.detectChanges();
|
||||
// });
|
||||
|
||||
// it('should create', () => {
|
||||
// expect(component).toBeTruthy();
|
||||
// });
|
||||
// });
|
@ -0,0 +1,60 @@
|
||||
import { Component, Inject, OnInit } from '@angular/core';
|
||||
import { MAT_DIALOG_DATA, MatDialog, MatDialogRef } from '@angular/material/dialog';
|
||||
import { MqttRequestService } from 'src/app/Service/Mqtt/mqtt-request.service';
|
||||
import { Plant } from 'src/app/models/plant.model';
|
||||
import { ErrorDialogComponent } from '../error-dialog/error-dialog.component';
|
||||
|
||||
@Component({
|
||||
selector: 'app-configure-plant-dialog',
|
||||
templateUrl: './configure-plant-dialog.component.html',
|
||||
styleUrls: ['./configure-plant-dialog.component.css']
|
||||
})
|
||||
export class ConfigurePlantDialogComponent implements OnInit {
|
||||
plant: Plant;
|
||||
|
||||
constructor(public dialogRef: MatDialogRef<ConfigurePlantDialogComponent>,
|
||||
@Inject(MAT_DIALOG_DATA) public data: any,
|
||||
private mqttRequestService: MqttRequestService,
|
||||
public dialog: MatDialog) {
|
||||
this.plant = data.plant;
|
||||
}
|
||||
|
||||
ngOnInit(): void {
|
||||
}
|
||||
|
||||
onSaveDialog() {
|
||||
this.plant.AirTemperature = +(<HTMLInputElement>document.getElementById("inputAirTemperature")).value;
|
||||
this.plant.AirHumidity = +(<HTMLInputElement>document.getElementById("inputAirHumidity")).value;
|
||||
this.plant.SoilMoisture = +(<HTMLInputElement>document.getElementById("inputSoilMoisture")).value;
|
||||
this.plant.Brightness = +(<HTMLInputElement>document.getElementById("inputBrightness")).value;
|
||||
|
||||
const plantInputString: string = this.plantInputTestFails();
|
||||
if (plantInputString) {
|
||||
this.dialog.open(ErrorDialogComponent, {
|
||||
data: "Error: " + plantInputString,
|
||||
width: '400px',
|
||||
});
|
||||
} else {
|
||||
//Request to the Backend, to change the data of the plant
|
||||
this.mqttRequestService.publishToCONFIGUREPLANT(this.plant);
|
||||
this.mqttRequestService.publishToGETALLDATA();
|
||||
window.location.reload();
|
||||
}
|
||||
}
|
||||
|
||||
plantInputTestFails(): string {
|
||||
if (this.plant.AirTemperature > 100) {
|
||||
return "Air Temperature is to high";
|
||||
}
|
||||
if (this.plant.AirHumidity > 100) {
|
||||
return "Air Humidity is to high";
|
||||
}
|
||||
if (this.plant.SoilMoisture > 100) {
|
||||
return "Soil Moisture is to high";
|
||||
}
|
||||
if (this.plant.Brightness > 100) {
|
||||
return "Brightness is to high";
|
||||
}
|
||||
return ""; //Empty string is a Boolean false!
|
||||
}
|
||||
}
|
@ -0,0 +1,10 @@
|
||||
:host ::ng-deep {
|
||||
.mat-dialog-title {
|
||||
color: red;
|
||||
}
|
||||
|
||||
.mat-button {
|
||||
background-color: ghostwhite;
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,8 @@
|
||||
<div class="container">
|
||||
<p mat-dialog-title>Delete Plant: {{plant.PlantName}}</p>
|
||||
<p mat-dialog-content>Soll die Pflanze "{{plant.PlantName}}" tatsächlich gelöscht werden?</p>
|
||||
<div mat-dialog-actions>
|
||||
<button mat-button mat-dialog-close>No</button>
|
||||
<button mat-button mat-dialog-close cdkFocusInitial (click)="onDelete()" color="warn">Delete</button>
|
||||
</div>
|
||||
</div>
|
@ -0,0 +1,23 @@
|
||||
// import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { DeleteConfirmDialogComponent } from './delete-confirm-dialog.component';
|
||||
|
||||
// describe('DeleteConfirmDialogComponent', () => {
|
||||
// let component: DeleteConfirmDialogComponent;
|
||||
// let fixture: ComponentFixture<DeleteConfirmDialogComponent>;
|
||||
|
||||
// beforeEach(async () => {
|
||||
// await TestBed.configureTestingModule({
|
||||
// declarations: [ DeleteConfirmDialogComponent ]
|
||||
// })
|
||||
// .compileComponents();
|
||||
|
||||
// fixture = TestBed.createComponent(DeleteConfirmDialogComponent);
|
||||
// component = fixture.componentInstance;
|
||||
// fixture.detectChanges();
|
||||
// });
|
||||
|
||||
// it('should create', () => {
|
||||
// expect(component).toBeTruthy();
|
||||
// });
|
||||
// });
|
@ -0,0 +1,24 @@
|
||||
import { Component, Inject } from '@angular/core';
|
||||
import { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';
|
||||
import { MqttRequestService } from 'src/app/Service/Mqtt/mqtt-request.service';
|
||||
import { Plant } from 'src/app/models/plant.model';
|
||||
import { ConfigurePlantDialogComponent } from '../configure-plant-dialog/configure-plant-dialog.component';
|
||||
|
||||
@Component({
|
||||
selector: 'app-delete-confirm-dialog',
|
||||
templateUrl: './delete-confirm-dialog.component.html',
|
||||
styleUrls: ['./delete-confirm-dialog.component.css']
|
||||
})
|
||||
export class DeleteConfirmDialogComponent {
|
||||
plant: Plant;
|
||||
constructor(public dialogRef: MatDialogRef<ConfigurePlantDialogComponent>,
|
||||
@Inject(MAT_DIALOG_DATA) public data: Plant, private mqttRequestService: MqttRequestService) {
|
||||
this.plant = data;
|
||||
}
|
||||
|
||||
|
||||
onDelete() {
|
||||
this.mqttRequestService.publishToDELETEPLANT(this.plant.PlantID);
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,10 @@
|
||||
:host ::ng-deep {
|
||||
.mat-dialog-title {
|
||||
color: red;
|
||||
}
|
||||
|
||||
.mat-button {
|
||||
background-color: ghostwhite;
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,7 @@
|
||||
<div class="container">
|
||||
<p mat-dialog-title>Error</p>
|
||||
<p mat-dialog-content>{{text}}</p>
|
||||
<div mat-dialog-actions>
|
||||
<button mat-button mat-dialog-close (click)="onClose()">Close</button>
|
||||
</div>
|
||||
</div>
|
@ -0,0 +1,23 @@
|
||||
// import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { ErrorDialogComponent } from './error-dialog.component';
|
||||
|
||||
// describe('ErrorDialogComponent', () => {
|
||||
// let component: ErrorDialogComponent;
|
||||
// let fixture: ComponentFixture<ErrorDialogComponent>;
|
||||
|
||||
// beforeEach(async () => {
|
||||
// await TestBed.configureTestingModule({
|
||||
// declarations: [ ErrorDialogComponent ]
|
||||
// })
|
||||
// .compileComponents();
|
||||
|
||||
// fixture = TestBed.createComponent(ErrorDialogComponent);
|
||||
// component = fixture.componentInstance;
|
||||
// fixture.detectChanges();
|
||||
// });
|
||||
|
||||
// it('should create', () => {
|
||||
// expect(component).toBeTruthy();
|
||||
// });
|
||||
// });
|
@ -0,0 +1,25 @@
|
||||
import { Component, Inject, OnInit } from '@angular/core';
|
||||
import { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';
|
||||
import { ConfigurePlantDialogComponent } from '../configure-plant-dialog/configure-plant-dialog.component';
|
||||
|
||||
@Component({
|
||||
selector: 'app-error-dialog',
|
||||
templateUrl: './error-dialog.component.html',
|
||||
styleUrls: ['./error-dialog.component.css']
|
||||
})
|
||||
export class ErrorDialogComponent implements OnInit {
|
||||
|
||||
text: string = "";
|
||||
constructor(public dialogRef: MatDialogRef<ConfigurePlantDialogComponent>,
|
||||
@Inject(MAT_DIALOG_DATA) public data: any) {
|
||||
this.text = data;
|
||||
}
|
||||
|
||||
ngOnInit(): void {
|
||||
}
|
||||
|
||||
onClose() {
|
||||
window.location.reload();
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,27 @@
|
||||
.container {
|
||||
width: 18.75rem;
|
||||
height: 21.875rem;
|
||||
display: grid;
|
||||
grid-template-rows: 15% 70% 15%;
|
||||
justify-content: start;
|
||||
align-content: start;
|
||||
}
|
||||
|
||||
.item-header {
|
||||
grid-row: 1/2;
|
||||
justify-self: center;
|
||||
font-size: 1.6rem;
|
||||
color: red;
|
||||
}
|
||||
|
||||
.item-img {
|
||||
grid-row: 2/3;
|
||||
justify-self: center;
|
||||
width: 21.875rem;
|
||||
height: 15.625rem;
|
||||
}
|
||||
|
||||
.item-button {
|
||||
grid-row: 3/4;
|
||||
justify-self: center;
|
||||
}
|
@ -0,0 +1,7 @@
|
||||
<div class="container">
|
||||
<p class="item-header" mat-dialog-title>Map</p>
|
||||
<img class="item-img" src="assets/images/gruppe1.jpg" alt="not available">
|
||||
<div class="item-button" mat-dialog-actions>
|
||||
<button mat-button mat-dialog-close>Close</button>
|
||||
</div>
|
||||
</div>
|
@ -0,0 +1,23 @@
|
||||
// import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
|
||||
// import { MapPlantDialogComponent } from './map-plant-dialog.component';
|
||||
|
||||
// describe('MapPlantDialogComponent', () => {
|
||||
// let component: MapPlantDialogComponent;
|
||||
// let fixture: ComponentFixture<MapPlantDialogComponent>;
|
||||
|
||||
// beforeEach(async () => {
|
||||
// await TestBed.configureTestingModule({
|
||||
// declarations: [ MapPlantDialogComponent ]
|
||||
// })
|
||||
// .compileComponents();
|
||||
|
||||
// fixture = TestBed.createComponent(MapPlantDialogComponent);
|
||||
// component = fixture.componentInstance;
|
||||
// fixture.detectChanges();
|
||||
// });
|
||||
|
||||
// it('should create', () => {
|
||||
// expect(component).toBeTruthy();
|
||||
// });
|
||||
// });
|
@ -0,0 +1,15 @@
|
||||
import { Component, OnInit } from '@angular/core';
|
||||
|
||||
@Component({
|
||||
selector: 'app-map-plant-dialog',
|
||||
templateUrl: './map-plant-dialog.component.html',
|
||||
styleUrls: ['./map-plant-dialog.component.css']
|
||||
})
|
||||
export class MapPlantDialogComponent implements OnInit {
|
||||
|
||||
constructor() { }
|
||||
|
||||
ngOnInit(): void {
|
||||
}
|
||||
|
||||
}
|
10
software/Frontend/src/app/models/plant.model.ts
Normal file
@ -0,0 +1,10 @@
|
||||
export interface Plant {
|
||||
AirTemperature: number;
|
||||
AirHumidity: number;
|
||||
SoilMoisture: number;
|
||||
Brightness: number;
|
||||
PlantID: number;
|
||||
Timestamp: string;
|
||||
MeasurementID: string;
|
||||
PlantName: string;
|
||||
}
|
4
software/Frontend/src/app/models/plantCount.model.ts
Normal file
@ -0,0 +1,4 @@
|
||||
export interface PlantCount {
|
||||
CurrentCount: number;
|
||||
MaxCount: number;
|
||||
}
|
4
software/Frontend/src/app/models/position.model.ts
Normal file
@ -0,0 +1,4 @@
|
||||
export interface Position {
|
||||
Position: string,
|
||||
Timestamp: string
|
||||
}
|
4
software/Frontend/src/app/models/robotBattery.model.ts
Normal file
@ -0,0 +1,4 @@
|
||||
export interface RobotBattery {
|
||||
Battery: number;
|
||||
Timestamp: string;
|
||||
}
|
0
software/Frontend/src/assets/.gitkeep
Normal file
After Width: | Height: | Size: 13 KiB |
After Width: | Height: | Size: 10 KiB |
After Width: | Height: | Size: 15 KiB |
After Width: | Height: | Size: 5.7 KiB |
BIN
software/Frontend/src/assets/images/Icons-Card/id_icon.png
Normal file
After Width: | Height: | Size: 17 KiB |
BIN
software/Frontend/src/assets/images/Icons-Card/name_icon.png
Normal file
After Width: | Height: | Size: 13 KiB |
After Width: | Height: | Size: 8.7 KiB |
BIN
software/Frontend/src/assets/images/background.jpg
Normal file
After Width: | Height: | Size: 114 KiB |
BIN
software/Frontend/src/assets/images/gruppe1.jpg
Normal file
After Width: | Height: | Size: 257 KiB |
BIN
software/Frontend/src/assets/images/gruppe2.jpg
Normal file
After Width: | Height: | Size: 257 KiB |
BIN
software/Frontend/src/assets/images/logo.png
Normal file
After Width: | Height: | Size: 29 KiB |
35
software/Frontend/src/custom-theme.scss
Normal file
@ -0,0 +1,35 @@
|
||||
|
||||
// Custom Theming for Angular Material
|
||||
// For more information: https://material.angular.io/guide/theming
|
||||
@use '@angular/material' as mat;
|
||||
// Plus imports for other components in your app.
|
||||
|
||||
// Include the common styles for Angular Material. We include this here so that you only
|
||||
// have to load a single css file for Angular Material in your app.
|
||||
// Be sure that you only ever include this mixin once!
|
||||
@include mat.core();
|
||||
|
||||
// Define the palettes for your theme using the Material Design palettes available in palette.scss
|
||||
// (imported above). For each palette, you can optionally specify a default, lighter, and darker
|
||||
// hue. Available color palettes: https://material.io/design/color/
|
||||
$Frontend-primary: mat.define-palette(mat.$indigo-palette);
|
||||
$Frontend-accent: mat.define-palette(mat.$pink-palette, A200, A100, A400);
|
||||
|
||||
// The warn palette is optional (defaults to red).
|
||||
$Frontend-warn: mat.define-palette(mat.$red-palette);
|
||||
|
||||
// Create the theme object. A theme consists of configurations for individual
|
||||
// theming systems such as "color" or "typography".
|
||||
$Frontend-theme: mat.define-light-theme((
|
||||
color: (
|
||||
primary: $Frontend-primary,
|
||||
accent: $Frontend-accent,
|
||||
warn: $Frontend-warn,
|
||||
)
|
||||
));
|
||||
|
||||
// Include theme styles for core and each component used in your app.
|
||||
// Alternatively, you can import and @include the theme mixins for each component
|
||||
// that you are using.
|
||||
@include mat.all-component-themes($Frontend-theme);
|
||||
|
3
software/Frontend/src/environments/environment.prod.ts
Normal file
@ -0,0 +1,3 @@
|
||||
export const environment = {
|
||||
production: true
|
||||
};
|
16
software/Frontend/src/environments/environment.ts
Normal file
@ -0,0 +1,16 @@
|
||||
// This file can be replaced during build by using the `fileReplacements` array.
|
||||
// `ng build` replaces `environment.ts` with `environment.prod.ts`.
|
||||
// The list of file replacements can be found in `angular.json`.
|
||||
|
||||
export const environment = {
|
||||
production: false
|
||||
};
|
||||
|
||||
/*
|
||||
* For easier debugging in development mode, you can import the following file
|
||||
* to ignore zone related error stack frames such as `zone.run`, `zoneDelegate.invokeTask`.
|
||||
*
|
||||
* This import should be commented out in production mode because it will have a negative impact
|
||||
* on performance if an error is thrown.
|
||||
*/
|
||||
// import 'zone.js/plugins/zone-error'; // Included with Angular CLI.
|
BIN
software/Frontend/src/favicon.ico
Normal file
After Width: | Height: | Size: 948 B |
19
software/Frontend/src/index.html
Normal file
@ -0,0 +1,19 @@
|
||||
<!doctype html>
|
||||
<html lang="en">
|
||||
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<title>Frontend</title>
|
||||
<base href="/">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no">
|
||||
<link rel="icon" type="image/x-icon" href="favicon.ico">
|
||||
<link rel="preconnect" href="https://fonts.gstatic.com">
|
||||
<link href="https://fonts.googleapis.com/css2?family=Roboto:wght@300;400;500&display=swap" rel="stylesheet">
|
||||
<link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet">
|
||||
</head>
|
||||
|
||||
<body class="mat-typography">
|
||||
<app-root></app-root>
|
||||
</body>
|
||||
|
||||
</html>
|
12
software/Frontend/src/main.ts
Normal file
@ -0,0 +1,12 @@
|
||||
import { enableProdMode } from '@angular/core';
|
||||
import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';
|
||||
|
||||
import { AppModule } from './app/app.module';
|
||||
import { environment } from './environments/environment';
|
||||
|
||||
if (environment.production) {
|
||||
enableProdMode();
|
||||
}
|
||||
|
||||
platformBrowserDynamic().bootstrapModule(AppModule)
|
||||
.catch(err => console.error(err));
|
53
software/Frontend/src/polyfills.ts
Normal file
@ -0,0 +1,53 @@
|
||||
/**
|
||||
* This file includes polyfills needed by Angular and is loaded before the app.
|
||||
* You can add your own extra polyfills to this file.
|
||||
*
|
||||
* This file is divided into 2 sections:
|
||||
* 1. Browser polyfills. These are applied before loading ZoneJS and are sorted by browsers.
|
||||
* 2. Application imports. Files imported after ZoneJS that should be loaded before your main
|
||||
* file.
|
||||
*
|
||||
* The current setup is for so-called "evergreen" browsers; the last versions of browsers that
|
||||
* automatically update themselves. This includes recent versions of Safari, Chrome (including
|
||||
* Opera), Edge on the desktop, and iOS and Chrome on mobile.
|
||||
*
|
||||
* Learn more in https://angular.io/guide/browser-support
|
||||
*/
|
||||
|
||||
/***************************************************************************************************
|
||||
* BROWSER POLYFILLS
|
||||
*/
|
||||
|
||||
/**
|
||||
* By default, zone.js will patch all possible macroTask and DomEvents
|
||||
* user can disable parts of macroTask/DomEvents patch by setting following flags
|
||||
* because those flags need to be set before `zone.js` being loaded, and webpack
|
||||
* will put import in the top of bundle, so user need to create a separate file
|
||||
* in this directory (for example: zone-flags.ts), and put the following flags
|
||||
* into that file, and then add the following code before importing zone.js.
|
||||
* import './zone-flags';
|
||||
*
|
||||
* The flags allowed in zone-flags.ts are listed here.
|
||||
*
|
||||
* The following flags will work for all browsers.
|
||||
*
|
||||
* (window as any).__Zone_disable_requestAnimationFrame = true; // disable patch requestAnimationFrame
|
||||
* (window as any).__Zone_disable_on_property = true; // disable patch onProperty such as onclick
|
||||
* (window as any).__zone_symbol__UNPATCHED_EVENTS = ['scroll', 'mousemove']; // disable patch specified eventNames
|
||||
*
|
||||
* in IE/Edge developer tools, the addEventListener will also be wrapped by zone.js
|
||||
* with the following flag, it will bypass `zone.js` patch for IE/Edge
|
||||
*
|
||||
* (window as any).__Zone_enable_cross_context_check = true;
|
||||
*
|
||||
*/
|
||||
|
||||
/***************************************************************************************************
|
||||
* Zone JS is required by default for Angular itself.
|
||||
*/
|
||||
import 'zone.js'; // Included with Angular CLI.
|
||||
|
||||
|
||||
/***************************************************************************************************
|
||||
* APPLICATION IMPORTS
|
||||
*/
|
4
software/Frontend/src/styles.css
Normal file
@ -0,0 +1,4 @@
|
||||
/* You can add global styles to this file, and also import other style files */
|
||||
|
||||
html, body { height: 100%; }
|
||||
body { margin: 0; font-family: Roboto, "Helvetica Neue", sans-serif; }
|
26
software/Frontend/src/test.ts
Normal file
@ -0,0 +1,26 @@
|
||||
// This file is required by karma.conf.js and loads recursively all the .spec and framework files
|
||||
|
||||
import 'zone.js/testing';
|
||||
import { getTestBed } from '@angular/core/testing';
|
||||
import {
|
||||
BrowserDynamicTestingModule,
|
||||
platformBrowserDynamicTesting
|
||||
} from '@angular/platform-browser-dynamic/testing';
|
||||
|
||||
declare const require: {
|
||||
context(path: string, deep?: boolean, filter?: RegExp): {
|
||||
<T>(id: string): T;
|
||||
keys(): string[];
|
||||
};
|
||||
};
|
||||
|
||||
// First, initialize the Angular testing environment.
|
||||
getTestBed().initTestEnvironment(
|
||||
BrowserDynamicTestingModule,
|
||||
platformBrowserDynamicTesting(),
|
||||
);
|
||||
|
||||
// Then we find all the tests.
|
||||
const context = require.context('./', true, /\.spec\.ts$/);
|
||||
// And load the modules.
|
||||
context.keys().forEach(context);
|
15
software/Frontend/tsconfig.app.json
Normal file
@ -0,0 +1,15 @@
|
||||
/* To learn more about this file see: https://angular.io/config/tsconfig. */
|
||||
{
|
||||
"extends": "./tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"outDir": "./out-tsc/app",
|
||||
"types": []
|
||||
},
|
||||
"files": [
|
||||
"src/main.ts",
|
||||
"src/polyfills.ts"
|
||||
],
|
||||
"include": [
|
||||
"src/**/*.d.ts"
|
||||
]
|
||||
}
|
32
software/Frontend/tsconfig.json
Normal file
@ -0,0 +1,32 @@
|
||||
/* To learn more about this file see: https://angular.io/config/tsconfig. */
|
||||
{
|
||||
"compileOnSave": false,
|
||||
"compilerOptions": {
|
||||
"baseUrl": "./",
|
||||
"outDir": "./dist/out-tsc",
|
||||
"forceConsistentCasingInFileNames": true,
|
||||
"strict": false,
|
||||
"noImplicitOverride": true,
|
||||
"noPropertyAccessFromIndexSignature": true,
|
||||
"noImplicitReturns": true,
|
||||
"noFallthroughCasesInSwitch": true,
|
||||
"sourceMap": true,
|
||||
"declaration": false,
|
||||
"downlevelIteration": true,
|
||||
"experimentalDecorators": true,
|
||||
"moduleResolution": "node",
|
||||
"importHelpers": true,
|
||||
"target": "es2020",
|
||||
"module": "es2020",
|
||||
"lib": [
|
||||
"es2020",
|
||||
"dom"
|
||||
]
|
||||
},
|
||||
"angularCompilerOptions": {
|
||||
"enableI18nLegacyMessageIdFormat": false,
|
||||
"strictInjectionParameters": true,
|
||||
"strictInputAccessModifiers": true,
|
||||
"strictTemplates": true
|
||||
}
|
||||
}
|