21 Commits

Author SHA1 Message Date
Willem Dantuma
b191751e02 AW-3477 fmHasPackage check werkt niet correct voor vandaag 2022-02-02 18:33:05 +01:00
66e492d2f8 versionnumber
All checks were successful
FarmMaps/FarmMapsLib/pipeline/head This commit looks good
2022-01-28 16:20:17 +01:00
e944064053 cherrypick ac0fe962a0: aw3299 hide questionmark on mobile
Some checks failed
FarmMaps/FarmMapsLib/pipeline/head There was a failure building this commit
2022-01-28 16:16:26 +01:00
a1a4fc14ab AW-3441 better test to find real problem. fixed the getValidPackages function.
All checks were successful
FarmMaps/FarmMapsLib/pipeline/head This commit looks good
2022-01-27 23:20:20 +01:00
122563a0bd AW-3441 another fix...
All checks were successful
FarmMaps/FarmMapsLib/pipeline/head This commit looks good
2022-01-27 22:35:47 +01:00
33f322424e AW-3441 fix oopsie
All checks were successful
FarmMaps/FarmMapsLib/pipeline/head This commit looks good
2022-01-27 17:25:12 +01:00
519b81d1fd AW-3441 fix package validation
All checks were successful
FarmMaps/FarmMapsLib/pipeline/head This commit looks good
2022-01-27 17:07:24 +01:00
b7c80dfdd8 AW-3441 add valid packages selector
All checks were successful
FarmMaps/FarmMapsLib/pipeline/head This commit looks good
2022-01-27 14:07:02 +01:00
d89670f669 Merge branch 'hotfix/AW-3412-B'
All checks were successful
FarmMaps/FarmMapsLib/pipeline/head This commit looks good
2022-01-19 14:38:49 +01:00
d025e2e3f6 export GeometryThumbnailComponent 2022-01-19 14:36:34 +01:00
Willem Dantuma
57407d83d3 Merge branch 'hotfix/AW-3412'
All checks were successful
FarmMaps/FarmMapsLib/pipeline/head This commit looks good
2022-01-19 13:24:14 +01:00
Willem Dantuma
f83549b5af Update version 2022-01-19 13:22:55 +01:00
Willem Dantuma
132556da81 AW-3412 2022-01-19 13:21:44 +01:00
Willem Dantuma
c2350eec52 Fix configuration for local authentication 2022-01-19 11:56:30 +01:00
Willem Dantuma
c6d14e6c9c Export interface
All checks were successful
FarmMaps/FarmMapsLib/pipeline/head This commit looks good
2021-12-02 11:23:36 +01:00
Willem Dantuma
8e4364bd08 bump version
All checks were successful
FarmMaps/FarmMapsLib/pipeline/head This commit looks good
2021-12-02 11:16:43 +01:00
Willem Dantuma
af3340ed70 Inject clicked feature in component
Some checks failed
FarmMaps/FarmMapsLib/pipeline/head There was a failure building this commit
2021-12-02 11:13:36 +01:00
Willem Dantuma
ff19d830a7 Fix version
All checks were successful
FarmMaps/FarmMapsLib/pipeline/head This commit looks good
2021-11-26 12:06:17 +01:00
Willem Dantuma
29914cfb1b oops
Some checks reported errors
FarmMaps/FarmMapsLib/pipeline/head Something is wrong with the build of this commit
2021-11-26 12:04:17 +01:00
Willem Dantuma
7bca0a57c4 Fix build
Some checks failed
FarmMaps/FarmMapsLib/pipeline/head There was a failure building this commit
2021-11-26 12:02:01 +01:00
Willem Dantuma
fdba357e9c Fix matching
Some checks failed
FarmMaps/FarmMapsLib/pipeline/head There was a failure building this commit
2021-11-26 11:58:11 +01:00
31 changed files with 361 additions and 413 deletions

8
Jenkinsfile vendored
View File

@@ -8,13 +8,13 @@ pipeline {
stage('npm install'){ stage('npm install'){
steps { steps {
sh '''rm -rf node_modules/ sh '''rm -rf node_modules/
npm install npm install --legacy-peer-deps
cd projects/common cd projects/common
npm install npm install --legacy-peer-deps
cd ../common-map cd ../common-map
npm install npm install --legacy-peer-deps
cd ../common-map3d cd ../common-map3d
npm install npm install --legacy-peer-deps
''' '''
} }
} }

20
package-lock.json generated
View File

@@ -1,6 +1,6 @@
{ {
"name": "farmmaps-lib-app", "name": "farmmaps-lib-app",
"version": "1.1.1", "version": "1.1.13",
"lockfileVersion": 1, "lockfileVersion": 1,
"requires": true, "requires": true,
"dependencies": { "dependencies": {
@@ -4079,25 +4079,25 @@
"dev": true "dev": true
}, },
"@farmmaps/common": { "@farmmaps/common": {
"version": "1.1.1-prerelease.2031", "version": "1.1.12",
"resolved": "https://repository.akkerweb.nl/repository/npm-group/@farmmaps/common/-/common-1.1.1-prerelease.2031.tgz", "resolved": "https://repository.akkerweb.nl/repository/npm-group/@farmmaps/common/-/common-1.1.12.tgz",
"integrity": "sha512-lSmTd2D1/PqzaXBjjY4X6UTYA5joZndXyZpUkFNm/Z0TTFmzFKmiIyGRA9AHX4StdQXC7QsCNfalYN31tnwtuQ==", "integrity": "sha512-TqPyPv35wK4bM1D/ZnT7zmno12pIlmBVesMKMJF7VtddZUm9mZe890ZQv7JJrbgURhU1o4mBs2ytmW0dGhDArw==",
"requires": { "requires": {
"tslib": "^2.0.0" "tslib": "^2.0.0"
} }
}, },
"@farmmaps/common-map": { "@farmmaps/common-map": {
"version": "1.1.1-prerelease.2031", "version": "1.1.12",
"resolved": "https://repository.akkerweb.nl/repository/npm-group/@farmmaps/common-map/-/common-map-1.1.1-prerelease.2031.tgz", "resolved": "https://repository.akkerweb.nl/repository/npm-group/@farmmaps/common-map/-/common-map-1.1.12.tgz",
"integrity": "sha512-BQ2bvNohPth0+oxoSoh+WWLJ/mP2CGW4Qf4xqgC3tgfW+UZfD0dBPl7IT2piuvOuUQzlSVloinCJ4ciZGyZRXg==", "integrity": "sha512-mu2AaIrGEbfj7gzJZLwLwBawOWwz4cJXDxcrsZQRzTb8Z9BA3ZV+9w1XkGU7Q0CgYLPuGaWMFF6AHqzTPARj/g==",
"requires": { "requires": {
"tslib": "^2.0.0" "tslib": "^2.0.0"
} }
}, },
"@farmmaps/common-map3d": { "@farmmaps/common-map3d": {
"version": "1.1.1-prerelease.2031", "version": "1.1.12",
"resolved": "https://repository.akkerweb.nl/repository/npm-group/@farmmaps/common-map3d/-/common-map3d-1.1.1-prerelease.2031.tgz", "resolved": "https://repository.akkerweb.nl/repository/npm-group/@farmmaps/common-map3d/-/common-map3d-1.1.12.tgz",
"integrity": "sha512-2M5wv5sFwlVwPGlN1b1AGM3Dh/DJFS7+L3b6e6ir2+ASoyy7xhZ67IWS2du8B7y7/xVsgxCJcJp1EJezbSCMAA==", "integrity": "sha512-yHRrL97GQZj0m4AlhsNrnKL6aujCBr5qgbp7uiUqsSQXp2dpJXbUFZpPnPKvOrFDUNJl8gqsbo5cvsvHucqCYg==",
"requires": { "requires": {
"tslib": "^2.0.0" "tslib": "^2.0.0"
} }

View File

@@ -1,6 +1,6 @@
{ {
"name": "farmmaps-lib-app", "name": "farmmaps-lib-app",
"version": "1.1.1", "version": "1.1.13",
"scripts": { "scripts": {
"ng": "ng", "ng": "ng",
"start": "ng serve", "start": "ng serve",
@@ -19,9 +19,9 @@
"@angular/platform-browser": "~11.2.14", "@angular/platform-browser": "~11.2.14",
"@angular/platform-browser-dynamic": "~11.2.14", "@angular/platform-browser-dynamic": "~11.2.14",
"@angular/router": "~11.2.14", "@angular/router": "~11.2.14",
"@farmmaps/common": "1.1.1-prerelease.2031", "@farmmaps/common": "1.1.12",
"@farmmaps/common-map": "1.1.1-prerelease.2031", "@farmmaps/common-map": "1.1.12",
"@farmmaps/common-map3d": "1.1.1-prerelease.2031", "@farmmaps/common-map3d": "1.1.12",
"@microsoft/signalr": "^3.1.16", "@microsoft/signalr": "^3.1.16",
"@ng-bootstrap/ng-bootstrap": "^9.0", "@ng-bootstrap/ng-bootstrap": "^9.0",
"@ngrx/effects": "^11.0", "@ngrx/effects": "^11.0",

View File

@@ -19,6 +19,7 @@ import * as mapEffects from './effects/map.effects';
import { IMapState} from './models/map.state'; import { IMapState} from './models/map.state';
import { ISelectedFeatures } from './models/selected.features'; import { ISelectedFeatures } from './models/selected.features';
import { IItemLayer,ItemLayer,ITemporalItemLayer,TemporalItemLayer } from './models/item.layer'; import { IItemLayer,ItemLayer,ITemporalItemLayer,TemporalItemLayer } from './models/item.layer';
import { IClickedFeature } from './models/clicked.feature';
import { IPeriodState } from './models/period.state'; import { IPeriodState } from './models/period.state';
// components // components
@@ -83,7 +84,7 @@ export function LocalStorageSync(reducer: ActionReducer<any>): ActionReducer<any
r2["mapState"] = JSON.parse(ms); r2["mapState"] = JSON.parse(ms);
} }
let sp = window.localStorage.getItem(MODULE_NAME+"_searchPeriod"); let sp = window.localStorage.getItem(MODULE_NAME+"_searchPeriod");
if(sp) { if(sp) {
let p = JSON.parse(sp); let p = JSON.parse(sp);
r2["period"] = { startDate: new Date(Date.parse(p.startDate)),endDate:new Date(Date.parse(p.endDate))}; r2["period"] = { startDate: new Date(Date.parse(p.startDate)),endDate:new Date(Date.parse(p.endDate))};
} }
@@ -163,7 +164,9 @@ export {
ITemporalItemLayer, ITemporalItemLayer,
TemporalItemLayer, TemporalItemLayer,
ifZoomToShowDirective, ifZoomToShowDirective,
ZoomToShowAlert ZoomToShowAlert,
IClickedFeature,
GeometryThumbnailComponent
} }
@NgModule({ @NgModule({
@@ -273,7 +276,8 @@ export {
FeatureListFeatureContainerComponent, FeatureListFeatureContainerComponent,
ZoomToExtentComponent, ZoomToExtentComponent,
ifZoomToShowDirective, ifZoomToShowDirective,
ZoomToShowAlert ZoomToShowAlert,
GeometryThumbnailComponent
], ],
providers: [ providers: [
FeatureIconService, FeatureIconService,
@@ -296,5 +300,5 @@ export {
}) })
export class AppCommonMapModule { export class AppCommonMapModule {
} }

View File

@@ -38,7 +38,7 @@ export class FeatureListContainerComponent {
let criteria=0; let criteria=0;
if (this.featureLists[i]['forItemType']) { if (this.featureLists[i]['forItemType']) {
criteria++; criteria++;
if( this.featureLists[i]['forItemType'].indexOf(queryState.itemType) >= 0) { if( this.featureLists[i]['forItemType'].split(",").filter(part => part == queryState.itemType).length == 1) {
matches++; matches++;
} }
} }

View File

@@ -32,7 +32,7 @@ export class FeatureListFeatureContainerComponent {
let criteria=0; let criteria=0;
if (this.featureLists[i]['forItemType']) { if (this.featureLists[i]['forItemType']) {
criteria++; criteria++;
if(this.featureLists[i]['forItemType'].indexOf(this.feature.get("itemType")) >= 0) { if(this.featureLists[i]['forItemType'].split(",").filter(part => part == this.feature.get("itemType")).length == 1) {
matches++; matches++;
} }
} }

View File

@@ -1,4 +1,4 @@
import { Component, Input, OnInit,ViewChild } from '@angular/core'; import { Component, Input, AfterViewInit, ViewChild } from '@angular/core';
import { Feature} from 'ol'; import { Feature} from 'ol';
import { Geometry } from 'ol/geom'; import { Geometry } from 'ol/geom';
import * as extent from 'ol/extent'; import * as extent from 'ol/extent';
@@ -10,51 +10,86 @@ import * as style from 'ol/style';
templateUrl: './feature-thumbnail.component.html', templateUrl: './feature-thumbnail.component.html',
styleUrls: ['./feature-thumbnail.component.scss'] styleUrls: ['./feature-thumbnail.component.scss']
}) })
export class GeometryThumbnailComponent implements OnInit { export class GeometryThumbnailComponent implements AfterViewInit {
constructor() { } constructor() { }
@ViewChild('canvas') canvas; @ViewChild('canvas') canvas;
@ViewChild('container') container; @ViewChild('container') container;
@Input('feature') feature:Feature;
ngOnInit(): void { private geometry:Geometry = null;
@Input() set feature(value:Feature) {
if(value) {
this.geometry = value.getGeometry();
} else {
this.geometry = null;
}
this.render(this.canvas,
this.geometryStyle,
this.geometry,
this.width,
this.height);
};
private defaultStyle:style.Style = new style.Style({
stroke: new style.Stroke({ color: 'black',width:1.5 })
});
private geometryStyle:style.Style = this.defaultStyle;
@Input() set fillColor(value:string) {
if(style) {
this.geometryStyle = new style.Style({
stroke: new style.Stroke({ color: 'black',width:1.5 }),
fill: new style.Fill({color: value})
});
} else {
this.geometryStyle = this.defaultStyle
}
this.render(this.canvas,
this.geometryStyle,
this.geometry,
this.width,
this.height);
} }
render(canvas,width,height,geometry:Geometry) { private width:number = 0;
let renderContext = render.toContext(canvas.getContext( '2d'),{ size: [width, height] }); private height:number = 0;
let strokeStyle = new style.Style({
stroke: new style.Stroke({ color: 'black',width:1.5 })
});
let geom = geometry.clone(), render(canvas,style:style.Style,geometry:Geometry,width:number,height:number) {
line = geom.getCoordinates()[0], if(canvas && canvas.nativeElement && geometry && style) {
e = extent.boundingExtent( line ); let renderContext = render.toContext(canvas.nativeElement.getContext( '2d'),{ size: [width, height] });
let dxy = extent.getCenter(e), let geom = geometry.clone(),
sxy = [ line = geom.getCoordinates()[0],
(width - 2 ) / extent.getWidth(e), e = extent.boundingExtent( line );
(height - 2 ) / extent.getHeight(e)
]; let dxy = extent.getCenter(e),
sxy = [
let dx = dxy[0], (width - 2 ) / extent.getWidth(e),
dy = dxy[1], (height - 2 ) / extent.getHeight(e)
sx = sxy[0], ];
sy = sxy[1];
let dx = dxy[0],
geom.translate( -dx, -dy ); dy = dxy[1],
geom.scale( Math.min(sx, sy), -Math.min(sx, sy)); sx = sxy[0],
geom.translate( width / 2, height / 2 ); sy = sxy[1];
renderContext.setStyle( strokeStyle ); geom.translate( -dx, -dy );
renderContext.drawGeometry( geom ); geom.scale( Math.min(sx, sy), -Math.min(sx, sy));
geom.translate(width / 2,height / 2 );
renderContext.setStyle( style );
renderContext.drawGeometry( geom );
}
} }
ngAfterViewInit() { ngAfterViewInit() {
this.render(this.canvas.nativeElement, this.width = this.container.nativeElement.offsetWidth;
this.container.nativeElement.offsetWidth, this.height = this.container.nativeElement.offsetHeight;
this.container.nativeElement.offsetHeight, this.render(this.canvas,
this.feature.getGeometry()); this.geometryStyle,
this.geometry,
this.width,
this.height);
} }
} }

View File

@@ -33,13 +33,13 @@ export class ItemListItemContainerComponent {
let criteria=0; let criteria=0;
if (this.itemComponentList[i]['forItemType']) { if (this.itemComponentList[i]['forItemType']) {
criteria++; criteria++;
if(this.itemComponentList[i]['forItemType'].indexOf(this.item.itemType) >= 0) { if(this.itemComponentList[i]['forItemType'].split(",").filter(part => part ==this.item.itemType).length == 1) {
matches++; matches++;
} }
} }
if (this.itemComponentList[i]['forSourceTask']) { if (this.itemComponentList[i]['forSourceTask']) {
criteria++; criteria++;
if(this.itemComponentList[i]['forSourceTask'].indexOf(this.item.sourceTask) >= 0) { if(this.itemComponentList[i]['forSourceTask'].split(",").filter(part => part ==this.item.sourceTask).length ==1) {
matches++; matches++;
} }
} }

View File

@@ -74,7 +74,7 @@
</div> </div>
</fm-side-panel> </fm-side-panel>
<fm-side-panel [resizeable]="true" [visible]="!noContent"> <fm-side-panel [resizeable]="true" [visible]="!noContent">
<router-outlet></router-outlet> <router-outlet (activate)="handleSidepaneloutletActivate($event)" (deactivate)="handleSidepaneloutletDeactivate($event)"></router-outlet>
</fm-side-panel> </fm-side-panel>
</div> </div>
</aol-map> </aol-map>

View File

@@ -11,6 +11,7 @@ import { DeviceService } from '@farmmaps/common';
import * as mapReducers from '../../reducers/map.reducer'; import * as mapReducers from '../../reducers/map.reducer';
import * as mapActions from '../../actions/map.actions'; import * as mapActions from '../../actions/map.actions';
import { IMapState} from '../../models/map.state'; import { IMapState} from '../../models/map.state';
import { IClickedFeature} from '../../models/clicked.feature';
import { IQuery } from '../../reducers/map.reducer' import { IQuery } from '../../reducers/map.reducer'
import { ISelectedFeatures } from '../../models/selected.features'; import { ISelectedFeatures } from '../../models/selected.features';
import { IItemLayer } from '../../models/item.layer'; import { IItemLayer } from '../../models/item.layer';
@@ -175,6 +176,18 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit {
this.store.dispatch(new mapActions.DoQuery(queryState)); this.store.dispatch(new mapActions.DoQuery(queryState));
} }
handleSidepaneloutletActivate(component:any) {
if(component && component.hasOwnProperty('clickedFeature')) {
(component as IClickedFeature).clickedFeature = this.clickedFeature;
}
}
handleSidepaneloutletDeactivate(component:any) {
if(component && component.hasOwnProperty('clickedFeature')) {
(component as IClickedFeature).clickedFeature = null;
}
}
ngOnInit() { ngOnInit() {
this.initialized = false; this.initialized = false;
console.debug("Init"); console.debug("Init");

View File

@@ -34,13 +34,13 @@ export class SelectedItemContainerComponent {
let criteria=0; let criteria=0;
if (this.selectedItemComponents[i]['forItemType'] ) { if (this.selectedItemComponents[i]['forItemType'] ) {
criteria++; criteria++;
if(this.selectedItemComponents[i]['forItemType'].indexOf(this.item.itemType) >= 0) { if(this.selectedItemComponents[i]['forItemType'].split(",").filter(part => part ==this.item.itemType).length == 1) {
matches++; matches++;
} }
} }
if (this.selectedItemComponents[i]['forSourceTask']) { if (this.selectedItemComponents[i]['forSourceTask']) {
criteria++; criteria++;
if( this.selectedItemComponents[i]['forSourceTask'].indexOf(this.item.sourceTask) >= 0) { if( this.selectedItemComponents[i]['forSourceTask'].split(",").filter(part => part ==this.item.sourceTask).length == 1) {
matches++; matches++;
} }
} }

View File

@@ -0,0 +1,10 @@
import {Feature} from 'ol';
import {Geometry} from 'ol/geom';
import { Observable } from 'rxjs';
/**
* @deprecated This interface will be removed soon
*/
export interface IClickedFeature {
clickedFeature: Observable<Feature<Geometry>>
}

View File

@@ -1,25 +0,0 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { AppMenuComponent } from './app-menu.component';
describe('AppMenuComponent', () => {
let component: AppMenuComponent;
let fixture: ComponentFixture<AppMenuComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
declarations: [ AppMenuComponent ]
})
.compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(AppMenuComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@@ -1,25 +0,0 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { AvatarComponent } from './avatar.component';
describe('AvatarComponent', () => {
let component: AvatarComponent;
let fixture: ComponentFixture<AvatarComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
declarations: [ AvatarComponent ]
})
.compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(AvatarComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@@ -1,25 +0,0 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { EditImageModalComponent } from './edit-image-modal.component';
describe('EditImageModalComponent', () => {
let component: EditImageModalComponent;
let fixture: ComponentFixture<EditImageModalComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
declarations: [ EditImageModalComponent ]
})
.compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(EditImageModalComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@@ -1,25 +0,0 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { GradientSelectComponent } from './gradient-select.component';
describe('GradientSelectComponent', () => {
let component: GradientSelectComponent;
let fixture: ComponentFixture<GradientSelectComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
declarations: [ GradientSelectComponent ]
})
.compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(GradientSelectComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@@ -1,25 +0,0 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { GradientComponent } from './gradient.component';
describe('GradientComponent', () => {
let component: GradientComponent;
let fixture: ComponentFixture<GradientComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
declarations: [ GradientComponent ]
})
.compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(GradientComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@@ -1,61 +1,61 @@
import { Directive, ViewContainerRef,TemplateRef,OnInit,Input,OnDestroy } from '@angular/core'; import {AfterViewInit, Directive, Input, OnDestroy, TemplateRef, ViewContainerRef} from '@angular/core';
import { Store} from '@ngrx/store'; import {Subscription} from 'rxjs';
import * as appCommonReducer from '../../reducers/app-common.reducer' import {PackageService} from '../../services/package.service';
import { IPackages } from '../../models/package'; import * as appCommonReducer from '../../reducers/app-common.reducer';
import { Observable, Subscription } from 'rxjs'; import {Store} from '@ngrx/store';
@Directive({ @Directive({
selector: '[fmHasPackage]', selector: '[fmHasPackage]',
}) })
export class HasPackageDirective implements OnDestroy{ export class HasPackageDirective implements OnDestroy, AfterViewInit {
@Input() @Input()
set fmHasPackage(packageIdentifier:string) { set fmHasPackage(packageIdentifier: string) {
this.packageIdentifier$ = packageIdentifier; this.packageIdentifier = packageIdentifier;
this.updateView(); this.updateView();
} }
@Input() @Input()
set fmHasPackageThen(thenTemplate:TemplateRef<any>) { set fmHasPackageThen(thenTemplate: TemplateRef<any>) {
this.thenTemplate$ = thenTemplate; this.thenTemplate = thenTemplate;
this.updateView(); this.updateView();
} }
@Input() @Input()
set fmHasPackageElse(thenTemplate:TemplateRef<any>) { set fmHasPackageElse(thenTemplate: TemplateRef<any>) {
this.elseTemplate$ = thenTemplate; this.elseTemplate = thenTemplate;
this.updateView(); this.updateView();
} }
private packageIdentifier$:string; private packageIdentifier: string;
private thenTemplate$:TemplateRef<any>; private thenTemplate: TemplateRef<any>;
private elseTemplate$:TemplateRef<any>; private elseTemplate: TemplateRef<any>;
private packages$:any = {}; private packSub: Subscription;
private packagesObservable$:Observable<IPackages> = this.store$.select(appCommonReducer.SelectGetUserPackages);
private packSub:Subscription;
constructor(private templateRef$: TemplateRef<any>,private viewContainerRef$: ViewContainerRef,private store$: Store<appCommonReducer.State>) { constructor(private hostTemplateRef: TemplateRef<any>, private viewContainerRef: ViewContainerRef,
this.thenTemplate$=templateRef$; private store: Store<appCommonReducer.State>, private packageService: PackageService) {
this.packSub = this.store$.select(appCommonReducer.SelectGetUserPackages).subscribe((packages) => { this.thenTemplate = hostTemplateRef;
this.packages$ = packages; this.packSub = this.store.select(appCommonReducer.SelectGetValidUserPackages).subscribe((_) => {
this.updateView(); this.updateView();
}); });
this.updateView();
} }
ngAfterViewInit() {
this.updateView();
}
updateView() { updateView() {
this.viewContainerRef$.clear(); this.viewContainerRef.clear();
const today = new Date(new Date(Date.now()).toUTCString()).setHours(0, 0, 0, 0); if (this.packageService.hasPackage(this.packageIdentifier)) {
if (this.packages$[this.packageIdentifier$] && this.viewContainerRef.createEmbeddedView(this.thenTemplate);
(this.packages$[this.packageIdentifier$].dataDate && new Date(this.packages$[this.packageIdentifier$].dataDate).setHours(0, 0, 0, 0) <= today) && } else if (this.elseTemplate) {
(this.packages$[this.packageIdentifier$].dataEndDate == null || new Date(this.packages$[this.packageIdentifier$].dataEndDate).setHours(0, 0, 0, 0) >= today)) { this.viewContainerRef.createEmbeddedView(this.elseTemplate);
this.viewContainerRef$.createEmbeddedView(this.thenTemplate$); }
} else if (this.elseTemplate$) {
this.viewContainerRef$.createEmbeddedView(this.elseTemplate$);
}
} }
ngOnDestroy() { ngOnDestroy() {
if(this.packSub) this.packSub.unsubscribe(); if (this.packSub) {
this.packSub.unsubscribe();
}
} }
} }

View File

@@ -1,4 +1,4 @@
<div> <div class="mobile-hide">
<div (click)="toggle($event)" class="rounded-circle menu-button hidden" [ngClass]="{'hidden':!user || noContent}"> <div (click)="toggle($event)" class="rounded-circle menu-button hidden" [ngClass]="{'hidden':!user || noContent}">
<span i18n-title title="Help"><i class="fas fa-question" aria-hidden="true"></i></span> <span i18n-title title="Help"><i class="fas fa-question" aria-hidden="true"></i></span>
<div class="menu hidden" [ngClass]="{'hidden':!showMenu}"> <div class="menu hidden" [ngClass]="{'hidden':!showMenu}">

View File

@@ -68,6 +68,16 @@ div.menu-button > span {
} }
} }
.mobile-hide {
display: inherit;
}
@media screen and (max-width: 768px) {
.mobile-hide {
display: none;
}
}
.unread { .unread {
display: block; display: block;
position: absolute; position: absolute;

View File

@@ -1,25 +0,0 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { HelpMenuComponent } from './help-menu.component';
describe('HelpMenuComponent', () => {
let component: HelpMenuComponent;
let fixture: ComponentFixture<HelpMenuComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
declarations: [ HelpMenuComponent ]
})
.compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(HelpMenuComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@@ -1,25 +0,0 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { NotificationMenuComponent } from './notification-menu.component';
describe('NotificationMenuComponent', () => {
let component: NotificationMenuComponent;
let fixture: ComponentFixture<NotificationMenuComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
declarations: [ NotificationMenuComponent ]
})
.compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(NotificationMenuComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@@ -1,25 +0,0 @@
import { async, ComponentFixture, TestBed } from '@angular/core/testing';
import { UserMenuComponent } from './user-menu.component';
describe('UserMenuComponent', () => {
let component: UserMenuComponent;
let fixture: ComponentFixture<UserMenuComponent>;
beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [ UserMenuComponent ]
})
.compileComponents();
}));
beforeEach(() => {
fixture = TestBed.createComponent(UserMenuComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@@ -7,5 +7,5 @@ export interface IPackage {
} }
export interface IPackages { export interface IPackages {
[id: string]: IPackage; [id: string]: IPackage[];
} }

View File

@@ -8,6 +8,7 @@ import { createSelector, createFeatureSelector, ActionReducerMap } from '@ngrx/s
import { MODULE_NAME } from '../module-name'; import { MODULE_NAME } from '../module-name';
import { IItem } from '../models/item'; import { IItem } from '../models/item';
import {getValidPackages, isValidPackage} from '../services/package.service';
export interface State { export interface State {
openedModalName: string, openedModalName: string,
@@ -66,7 +67,7 @@ export function reducer(state = initialState, action: appCommonActions.Actions )
name:a.userinfo["name"], name:a.userinfo["name"],
claims:claims, claims:claims,
searchable: false searchable: false
}; };
return tassign(state, { user: user }); return tassign(state, { user: user });
} }
case appCommonActions.INITROOTSUCCESS: { case appCommonActions.INITROOTSUCCESS: {
@@ -75,10 +76,10 @@ export function reducer(state = initialState, action: appCommonActions.Actions )
} }
case appCommonActions.OPENMODAL: { case appCommonActions.OPENMODAL: {
return tassign(state, { openedModalName: action.modalName }); return tassign(state, { openedModalName: action.modalName });
} }
case appCommonActions.CLOSEMODAL: { case appCommonActions.CLOSEMODAL: {
return tassign(state, { openedModalName: null }); return tassign(state, { openedModalName: null });
} }
case appCommonActions.LOADITEMTYPESSUCCESS: { case appCommonActions.LOADITEMTYPESSUCCESS: {
let a = action as appCommonActions.LoadItemTypesSuccess; let a = action as appCommonActions.LoadItemTypesSuccess;
return tassign(state, { itemTypes: a.itemTypes }); return tassign(state, { itemTypes: a.itemTypes });
@@ -131,7 +132,10 @@ export function reducer(state = initialState, action: appCommonActions.Actions )
a.items.forEach((item) => { a.items.forEach((item) => {
item.data.dataDate = item.dataDate; item.data.dataDate = item.dataDate;
item.data.dataEndDate = item.dataEndDate; item.data.dataEndDate = item.dataEndDate;
packages[item.data.id]=item.data; if (!packages[item.data.id]) {
packages[item.data.id] = [];
}
packages[item.data.id].push(item.data);
}); });
return tassign(state,{userPackages:packages}); return tassign(state,{userPackages:packages});
@@ -203,6 +207,9 @@ export const selectGetRouteLoading = createSelector(selectAppCommonState, getRou
export const SelectGetMenuVisible = createSelector(selectAppCommonState,getMenuVisible); export const SelectGetMenuVisible = createSelector(selectAppCommonState,getMenuVisible);
export const SelectGetUser = createSelector(selectAppCommonState,getUser); export const SelectGetUser = createSelector(selectAppCommonState,getUser);
export const SelectGetUserPackages = createSelector(selectAppCommonState,getUserPackages); export const SelectGetUserPackages = createSelector(selectAppCommonState,getUserPackages);
export const SelectGetValidUserPackages = createSelector(SelectGetUserPackages, (packageMap) => {
return getValidPackages(packageMap);
});
export const SelectGetUserSettingsRoot = createSelector(selectAppCommonState,getUserSettingsRoot); export const SelectGetUserSettingsRoot = createSelector(selectAppCommonState,getUserSettingsRoot);
export const SelectGetAccountMenuVisible = createSelector(selectAppCommonState,getAccountMenuVisible); export const SelectGetAccountMenuVisible = createSelector(selectAppCommonState,getAccountMenuVisible);
export const SelectGetAppMenuVisible = createSelector(selectAppCommonState,getAppMenuVisible); export const SelectGetAppMenuVisible = createSelector(selectAppCommonState,getAppMenuVisible);
@@ -213,3 +220,4 @@ export const SelectgetUnreadNotifications = createSelector(selectAppCommonState,
export const SelectGetIsOnline = createSelector(selectAppCommonState,getIsOnline); export const SelectGetIsOnline = createSelector(selectAppCommonState,getIsOnline);
export const SelectGetIsPageMode = createSelector(selectAppCommonState,getIsPageMode); export const SelectGetIsPageMode = createSelector(selectAppCommonState,getIsPageMode);

View File

@@ -0,0 +1,122 @@
import {getValidPackages, isValidPackage, PackageService} from './package.service';
import {MockStore, provideMockStore} from '@ngrx/store/testing';
import {TestBed} from '@angular/core/testing';
import * as appCommonReducer from '../reducers/app-common.reducer';
import {reducer, State} from '../reducers/app-common.reducer';
import * as appCommonActions from '../actions/app-common.actions';
import {IItem} from '../models/item';
import {IPackage} from '../models/package';
describe('PackageService', () => {
const initialState = {userPackages: {}} as State;
let serviceUnderTest: PackageService;
let items;
beforeEach(() => {
TestBed.configureTestingModule({
providers: [
provideMockStore({
initialState,
selectors: [
{
selector: appCommonReducer.SelectGetValidUserPackages,
value: {
'vnd.farmmaps.package.zoning': {
id: 'vnd.farmmaps.package.zoning',
name: 'zoning package',
dataDate: '2022-01-25T00:00:00.000000Z'
}
}
}
]
}),
]
});
const store = TestBed.inject(MockStore);
serviceUnderTest = new PackageService(store, null, null);
items = [
{
code: '123132',
dataDate: '2021-01-25T00:00:00Z',
dataEndDate: '2021-05-25T00:00:00Z',
data: {
id: 'vnd.farmmaps.package.zoning'
}
} as IItem,
{
code: '1231325',
dataDate: '2021-01-25T00:00:00Z',
dataEndDate: null,
data: {
id: 'vnd.farmmaps.package.zoning'
}
} as IItem,
{
code: '1231325',
dataDate: '2022-01-25T00:00:00Z',
dataEndDate: '2022-05-25T00:00:00Z',
data: {
id: 'vnd.farmmaps.package.tipstar'
}
} as IItem,
{
code: '1231325',
dataDate: '2022-01-25T00:00:00Z',
dataEndDate: null,
data: {
id: 'vnd.farmmaps.package.weather'
}
} as IItem
];
});
it('Does not have a tipstar package', () => {
const hasPackage = serviceUnderTest.hasPackage('vnd.farmmaps.package.tipstar');
expect(hasPackage).toBe(false);
});
it('Does have a zoning package', () => {
const hasPackage = serviceUnderTest.hasPackage('vnd.farmmaps.package.zoning');
expect(hasPackage).toBe(true);
});
it('reducer.INITUSERPACKAGESSUCCESS alter state correctly', () => {
const action = new appCommonActions.InitUserPackagesSuccess(items);
expect(Object.keys(initialState.userPackages).length).toBe(0);
const state = reducer(initialState, action);
expect(Object.keys(state.userPackages).length).toBe(3);
expect(state.userPackages['vnd.farmmaps.package.zoning'].length).toBe(2);
expect(state.userPackages['vnd.farmmaps.package.tipstar'].length).toBe(1);
expect(state.userPackages['vnd.farmmaps.package.weather'].length).toBe(1);
});
it('to validatePackage', () => {
const action = new appCommonActions.InitUserPackagesSuccess(items);
const state = reducer(initialState, action);
expect(isValidPackage(null)).toBe(false);
expect(isValidPackage(state.userPackages['vnd.farmmaps.package.tipstar'][0])).toBe(true);
expect(isValidPackage(state.userPackages['vnd.farmmaps.package.zoning'][0])).toBe(false);
expect(isValidPackage(state.userPackages['vnd.farmmaps.package.zoning'][1])).toBe(true);
expect(isValidPackage(state.userPackages['vnd.farmmaps.package.weather'][0])).toBe(true);
});
it('to getValidPackages', () => {
const action = new appCommonActions.InitUserPackagesSuccess(items);
const state = reducer(initialState, action);
const validPackages = getValidPackages(state.userPackages);
expect(Object.keys(validPackages).length).toBe(3);
// today is 27-01-2022, so the below package is not valid
expect(validPackages['vnd.farmmaps.package.zoning']).not.toEqual({
id: 'vnd.farmmaps.package.zoning',
dataDate: '2021-01-25T00:00:00Z',
dataEndDate: '2021-05-25T00:00:00Z'
} as unknown as IPackage);
});
});

View File

@@ -1,37 +1,58 @@
import { Injectable } from '@angular/core'; import {Injectable} from '@angular/core';
import { Store} from '@ngrx/store'; import {Store} from '@ngrx/store';
import * as appCommonReducer from '../reducers/app-common.reducer' import * as appCommonReducer from '../reducers/app-common.reducer';
import { IPackages } from '../models/package'; import {IPackage, IPackages} from '../models/package';
import { IItem } from '../models/item'; import {IItem} from '../models/item';
import { IItemTask } from '../models/itemTask'; import {IItemTask} from '../models/itemTask';
import { HttpClient } from "@angular/common/http"; import {HttpClient} from '@angular/common/http';
import { AppConfig } from "../shared/app.config"; import {AppConfig} from '../shared/app.config';
import { Observable } from 'rxjs'; import {Observable} from 'rxjs';
@Injectable({ @Injectable({
providedIn: 'root', providedIn: 'root',
}) })
export class PackageService { export class PackageService {
private packages$:IPackages = {}; private packages: { [key: string]: IPackage } = {};
constructor(private store$: Store<appCommonReducer.State>, public httpClient: HttpClient, public appConfig: AppConfig) { constructor(private store$: Store<appCommonReducer.State>, public httpClient: HttpClient, public appConfig: AppConfig) {
store$.select(appCommonReducer.SelectGetUserPackages).subscribe((packages) => { store$.select(appCommonReducer.SelectGetValidUserPackages).subscribe((packages) => {
this.packages$ = packages; this.packages = packages;
}) });
} }
ApiEndpoint() { ApiEndpoint() {
return this.appConfig.getConfig("apiEndPoint"); return this.appConfig.getConfig('apiEndPoint');
} }
hasPackage(id:string):boolean {
if(!this.packages$[id]) return false;
return this.packages$[id].enabled ? this.packages$[id].enabled == true : true;
}
postItemPackageTask(item: IItem, task: IItemTask): Observable<IItemTask> { hasPackage(id: string): boolean {
return this.httpClient.post<IItemTask>(`${this.ApiEndpoint()}/api/v1/items/${item.code}/packagetasks`, task); return id in this.packages;
} }
}
postItemPackageTask(item: IItem, task: IItemTask): Observable<IItemTask> {
return this.httpClient.post<IItemTask>(`${this.ApiEndpoint()}/api/v1/items/${item.code}/packagetasks`, task);
}
}
export function getValidPackages(packageMap: IPackages): {[key: string]: IPackage} {
const keys = Object.keys(packageMap);
return keys.filter(k => {
const packages = packageMap[k]
.filter((p) => isValidPackage(p));
return packages.length > 0;
}).reduce((map, key) => {
const packages = packageMap[key];
const newMap = {...map};
newMap[key] = packages.find(p => isValidPackage(p));
return newMap;
}, {});
}
export function isValidPackage(pack: IPackage): boolean {
const now = new Date(Date.now());
const utcToday = Date.UTC(now.getUTCFullYear(),now.getUTCMonth(),now.getUTCDate());
return pack !== null && new Date(pack.dataDate).getTime() <= utcToday
&& (!pack.dataEndDate || new Date(pack.dataEndDate).getTime() >= utcToday);
}

View File

@@ -1,25 +0,0 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { TestComponent } from './landingpage.component';
describe('TestComponent', () => {
let component: TestComponent;
let fixture: ComponentFixture<TestComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
declarations: [ TestComponent ]
})
.compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(TestComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@@ -1,25 +0,0 @@
import { async, ComponentFixture, TestBed } from '@angular/core/testing';
import { MenuComponent } from './menu.component';
describe('MenuComponent', () => {
let component: MenuComponent;
let fixture: ComponentFixture<MenuComponent>;
beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [ MenuComponent ]
})
.compileComponents();
}));
beforeEach(() => {
fixture = TestBed.createComponent(MenuComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@@ -1,25 +0,0 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { TestComponent } from './test.component';
describe('TestComponent', () => {
let component: TestComponent;
let fixture: ComponentFixture<TestComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
declarations: [ TestComponent ]
})
.compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(TestComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@@ -1,8 +1,8 @@
{ {
"issuer": "https://accounts.test.farmmaps.eu", "issuer": "http://localhost:8094",
"clientId": "farmmapsdev", "clientId": "farmmapsdev",
"audience": "http://localhost:8082", "audience": "http://localhost:8082/",
"requireHttps": true, "requireHttps": false,
"apiEndPoint": "http://localhost:8082", "apiEndPoint": "http://localhost:8082",
"grantType":"code" "grantType":"code"
} }