From 64c02614baa597bf43ffa578eed1142b10eaa00e Mon Sep 17 00:00:00 2001 From: Willem Dantuma Date: Thu, 2 Jan 2020 18:11:02 +0100 Subject: [PATCH] Implement pan to, fix zooming on map change --- projects/common-map/package.json | 2 +- .../src/fm-map/common-map.module.ts | 4 + .../pan-to-location.component.html | 13 ++ .../pan-to-location.component.scss | 28 ++++ .../pan-to-location.component.ts | 84 +++++++++++ .../rotation-reset.component.ts | 108 +++++++------- .../fm-map/components/map/map.component.html | 139 ++++++++++-------- .../fm-map/components/map/map.component.ts | 120 ++++++++------- 8 files changed, 325 insertions(+), 173 deletions(-) create mode 100644 projects/common-map/src/fm-map/components/aol/pan-to-location/pan-to-location.component.html create mode 100644 projects/common-map/src/fm-map/components/aol/pan-to-location/pan-to-location.component.scss create mode 100644 projects/common-map/src/fm-map/components/aol/pan-to-location/pan-to-location.component.ts diff --git a/projects/common-map/package.json b/projects/common-map/package.json index 9bd65a7..1051958 100644 --- a/projects/common-map/package.json +++ b/projects/common-map/package.json @@ -15,6 +15,6 @@ "@ngrx/router-store": "^8.2", "@ngrx/store": "^8.2", "tassign": "^1.0.0", - "@farmmaps/common": ">=0.0.1-prerelease.90 <0.0.1" + "@farmmaps/common": ">=0.0.1-prerelease.102 <0.0.1" } } diff --git a/projects/common-map/src/fm-map/common-map.module.ts b/projects/common-map/src/fm-map/common-map.module.ts index a4e9774..acb8c3c 100644 --- a/projects/common-map/src/fm-map/common-map.module.ts +++ b/projects/common-map/src/fm-map/common-map.module.ts @@ -65,6 +65,7 @@ import { WidgetStatusComponent } from './components/widget-status/widget-status. import { ForChild} from './components/for-item/for-child.decorator'; import {ForItemType } from './components/for-item/for-itemtype.decorator'; import { ForSourceTask} from './components/for-item/for-sourcetask.decorator'; +import { PanToLocation} from './components/aol/pan-to-location/pan-to-location.component'; export { @@ -104,6 +105,7 @@ export { ItemWidgetListComponent, WidgetStatusComponent, GpsLocation, + PanToLocation, AbstractFeatureListComponent, AbstractFeatureListFeatureComponent, AbstractSelectedItemComponent, @@ -169,6 +171,7 @@ export { ItemWidgetListComponent, WidgetStatusComponent, GpsLocation, + PanToLocation ], entryComponents: [ FeatureListComponent, @@ -192,6 +195,7 @@ export { MetaDataModalComponent, MapComponent, GpsLocation, + PanToLocation, FeatureListFeatureComponent, FeatureListFeatureCropfieldComponent, FeatureListFeatureCroppingschemeComponent, diff --git a/projects/common-map/src/fm-map/components/aol/pan-to-location/pan-to-location.component.html b/projects/common-map/src/fm-map/components/aol/pan-to-location/pan-to-location.component.html new file mode 100644 index 0000000..486e0aa --- /dev/null +++ b/projects/common-map/src/fm-map/components/aol/pan-to-location/pan-to-location.component.html @@ -0,0 +1,13 @@ +
+ + + +
diff --git a/projects/common-map/src/fm-map/components/aol/pan-to-location/pan-to-location.component.scss b/projects/common-map/src/fm-map/components/aol/pan-to-location/pan-to-location.component.scss new file mode 100644 index 0000000..60ac0b3 --- /dev/null +++ b/projects/common-map/src/fm-map/components/aol/pan-to-location/pan-to-location.component.scss @@ -0,0 +1,28 @@ +@import "../../../_theme.scss"; +@import "~bootstrap/scss/bootstrap.scss"; + + +.gps-location { + display:block; + width:2.5em; + height:2.5em; + background-color: white; + background-size: contain; + margin-top:0.5em; +} + +.center, .tolerance, .border { + stroke-width: 0; +} + +.pan-to { + fill: #333333; +} + +.pan-to-centered { + fill: theme-color() +} + +.pan-to-disabled { + fill: #808080; +} \ No newline at end of file diff --git a/projects/common-map/src/fm-map/components/aol/pan-to-location/pan-to-location.component.ts b/projects/common-map/src/fm-map/components/aol/pan-to-location/pan-to-location.component.ts new file mode 100644 index 0000000..b709e64 --- /dev/null +++ b/projects/common-map/src/fm-map/components/aol/pan-to-location/pan-to-location.component.ts @@ -0,0 +1,84 @@ +import { Component, OnInit, Input, Host, OnChanges, SimpleChanges,ChangeDetectorRef } from '@angular/core'; +import { MapComponent } from 'ngx-openlayers'; +import {IMapState} from '../../../models/map.state' +import {View} from 'ol'; +import { fromLonLat } from 'ol/proj'; + + +@Component({ + selector: 'fm-map-pan-to-location', + templateUrl: './pan-to-location.component.html', + styleUrls: ['./pan-to-location.component.scss'] +}) +export class PanToLocation implements OnInit,OnChanges{ + + view: View; + map: MapComponent; + @Input() position: Position; + @Input() mapState: IMapState; + @Input() animate: boolean; + + constructor(@Host() map: MapComponent,private changeDetectorRef$: ChangeDetectorRef ) { + this.map = map; + } + + ngOnInit() { + this.view = this.map.instance.getView(); + this.view.on('change:center', () => { + this.changeDetectorRef$.detectChanges(); + }); + } + + ngOnChanges(changes: SimpleChanges) { + // if (changes.position && this.instance) { + // var p = changes.position.currentValue as Position; + // this.instance.setPosition(fromLonLat([p.coords.longitude, p.coords.latitude])); + // this.locationTolerance = p.coords.accuracy; + // this.recalcLocationTolerance(); + // this.heading = p.coords.heading; + // } + } + + p + + public centered():boolean { + if(this.position && this.mapState) { + let center = this.view.getCenter(); + let newCenter = fromLonLat([this.position.coords.longitude,this.position.coords.latitude]); + let x1 = newCenter[0].toFixed(0); + let x2 = center[0].toFixed(0); + let y1 = newCenter[1].toFixed(0); + let y2 = center[1].toFixed(0); + return x1==x2 && y1==y2; + } + return false; + } + + public disabled():boolean { + return !this.position; + } + + handleClick(event:Event) { + if(this.position) { + let view = this.map.instance.getView(); + let newCenter = fromLonLat([this.position.coords.longitude,this.position.coords.latitude]); + let extent = [newCenter[0]-500,newCenter[1]-500,newCenter[0]+500,newCenter[1]+500]; + var options = { padding: [0, 0, 0, 0],minResolution:1 }; + let size = this.map.instance.getSize(); + let rem = parseFloat(getComputedStyle(document.documentElement).fontSize); + let threshold = 44 * rem; + var left = 1 * rem; + var right = 1 * rem; + var bottom = Math.round(size[1] / 2); + var top = 1 * rem; + if (size[0] > threshold) { + bottom = 1 * rem; + left = 23 * rem; + } + //options.padding = [top, right, bottom, left]; + if (this.animate) options["duration"] = 2000; + view.fit(extent, options); + } + event.preventDefault(); + } +} diff --git a/projects/common-map/src/fm-map/components/aol/rotation-reset/rotation-reset.component.ts b/projects/common-map/src/fm-map/components/aol/rotation-reset/rotation-reset.component.ts index 927ccd1..2902a3e 100644 --- a/projects/common-map/src/fm-map/components/aol/rotation-reset/rotation-reset.component.ts +++ b/projects/common-map/src/fm-map/components/aol/rotation-reset/rotation-reset.component.ts @@ -1,53 +1,55 @@ -import { Component, Host, Input, OnInit, ChangeDetectorRef } from '@angular/core'; -import { ViewComponent, MapComponent } from 'ngx-openlayers'; - -import {View} from 'ol'; - - - -@Component({ - selector: 'fm-map-rotation-reset', - template: `
`, - styles: [`.compass { - width:2.5em; - height:2.5em; - background-color: white; - background-size: contain; - background-image: url(); - opacity: 1; - } - .compass-n { - background-image: url(); - transition: opacity 1s ease-out; - transition-delay: 2s; - opacity:0; - } -`] -}) -export class RotationResetComponent implements OnInit { - view: View; - - public Rotation() { - let rotation = this.view ? this.view.getRotation() : 0; - return `rotate(${rotation}rad)`; - } - - public IsNorth() { - return this.view ? this.view.getRotation() == 0 : true; - } - - ngOnInit(): void { - this.view = this.map.instance.getView(); - this.view.on('change:rotation', () => { - this.changeDetectorRef$.detectChanges(); - }); - } - - constructor( @Host() private map: MapComponent, private changeDetectorRef$: ChangeDetectorRef ) { - } - - handleClick(event:Event) { - this.view.animate({ rotation: 0 }); - event.preventDefault(); - } -} +import { Component, Host, Input, OnInit, ChangeDetectorRef } from '@angular/core'; +import { ViewComponent, MapComponent } from 'ngx-openlayers'; + +import {View} from 'ol'; + + + +@Component({ + selector: 'fm-map-rotation-reset', + template: `
`, + styles: [`.compass { + width:2.5em; + height:2.5em; + background-color: white; + background-size: contain; + background-image: url(); + opacity: 1; + margin-top:0.5em; + } + .compass-n { + background-image: url(); + transition: opacity 1s ease-out 2s,height 1s ease-out 3s,margin-top 1s ease-out 3s; + opacity:0; + height:0; + margin-top:0; + } +`] +}) +export class RotationResetComponent implements OnInit { + view: View; + + public Rotation() { + let rotation = this.view ? this.view.getRotation() : 0; + return `rotate(${rotation}rad)`; + } + + public IsNorth() { + return this.view ? this.view.getRotation() == 0 : true; + } + + ngOnInit(): void { + this.view = this.map.instance.getView(); + this.view.on('change:rotation', () => { + this.changeDetectorRef$.detectChanges(); + }); + } + + constructor( @Host() private map: MapComponent, private changeDetectorRef$: ChangeDetectorRef ) { + } + + handleClick(event:Event) { + this.view.animate({ rotation: 0 }); + event.preventDefault(); + } +} diff --git a/projects/common-map/src/fm-map/components/map/map.component.html b/projects/common-map/src/fm-map/components/map/map.component.html index efe5cb2..39d1306 100644 --- a/projects/common-map/src/fm-map/components/map/map.component.html +++ b/projects/common-map/src/fm-map/components/map/map.component.html @@ -1,58 +1,81 @@ - -
- -
- - - - - - - - - - - - - -
- - -
- -
- - -
-
-
-
- -
- -
- -
- -
-
-
Cannot find {{(queryState|async)?.query}}
-
Cannot find tag {{(queryState|async)?.tags}}
-
-
-
-
- -
-
-
- -
-
-
- -
-
-
-
- + + +
+ +
+ + + + + + + + + + + + + +
+ + + +
+ +
+ + +
+
+
+
+ +
+ +
+ +
+ +
+
+
Cannot find {{state.queryState?.query}}
+
Cannot find tag {{state.queryState?.tags}}
+
+
+
+
+ +
+
+
+ +
+
+
+ +
+
+
+
+
+ + diff --git a/projects/common-map/src/fm-map/components/map/map.component.ts b/projects/common-map/src/fm-map/components/map/map.component.ts index 278eb71..6967667 100644 --- a/projects/common-map/src/fm-map/components/map/map.component.ts +++ b/projects/common-map/src/fm-map/components/map/map.component.ts @@ -40,39 +40,39 @@ import { tassign } from 'tassign'; export class MapComponent implements OnInit, OnDestroy,AfterViewInit { title: string = 'Map'; - public openedModalName: Observable; - public itemTypes: Observable<{ [id: string]: IItemType }>; - public mapState: Observable; - public features: Observable>; - public overlayLayers: Observable>; - public selectedOverlayLayer: Observable; - public selectedItemLayer: Observable; - public baseLayers: Observable>; - public selectedBaseLayer: Observable; - public projection: Observable; - public selectedFeatures: Subject = new Subject(); - public droppedFile: Subject = new Subject(); + public openedModalName$: Observable; + public itemTypes$: Observable<{ [id: string]: IItemType }>; + public mapState$: Observable; + public features$: Observable>; + public overlayLayers$: Observable>; + public selectedOverlayLayer$: Observable; + public selectedItemLayer$: Observable; + public baseLayers$: Observable>; + public selectedBaseLayer$: Observable; + public projection$: Observable; + public selectedFeatures$: Subject = new Subject(); + public droppedFile$: Subject = new Subject(); private paramSub: Subscription; private itemTypeSub: Subscription; private mapStateSub: Subscription; private queryStateSub: Subscription; - public parentCode: Observable; - public panelVisible: Observable; - public panelCollapsed: Observable; - public selectedFeature: Observable; - public selectedItem: Observable; - public queryState: Observable; - public period: Observable; - public clearEnabled: Observable; - public searchCollapsed: Observable; - public searchMinified: Observable; - public menuVisible: Observable; - public query: Observable; - public position: Observable; + public parentCode$: Observable; + public panelVisible$: Observable; + public panelCollapsed$: Observable; + public selectedFeature$: Observable; + public selectedItem$: Observable; + public queryState$: Observable; + public period$: Observable; + public clearEnabled$: Observable; + public searchCollapsed$: Observable; + public searchMinified$: Observable; + public menuVisible$: Observable; + public query$: Observable; + public position$: Observable; public baseLayersCollapsed:boolean = true; public overlayLayersCollapsed: boolean = true; - public extent: Observable; - @ViewChild('map', { static: true }) map; + public extent$: Observable; + @ViewChild('map', { static: false }) map; constructor(private store: Store, private route: ActivatedRoute, @@ -81,8 +81,7 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { private serializeService: StateSerializerService, public itemTypeService: ItemTypeService, private location: Location, - private geolocationService: GeolocationService, - private _ref: ChangeDetectorRef ) { + private geolocationService: GeolocationService) { } @HostListener('document:keyup', ['$event']) @@ -118,36 +117,36 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { ngOnInit() { this.store.dispatch(new mapActions.Init()); - this.selectedFeatures.next({x:0,y:0,features:[]}); - this.mapState = this.store.select(mapReducers.selectGetMapState); - this.parentCode = this.store.select(mapReducers.selectGetParentCode); - this.features = this.store.select(mapReducers.selectGetFeatures); - this.overlayLayers = this.store.select(mapReducers.selectGetOverlayLayers); - this.selectedOverlayLayer = this.store.select(mapReducers.selectGetSelectedOverlayLayer); - this.baseLayers = this.store.select(mapReducers.selectGetBaseLayers); - this.projection = this.store.select(mapReducers.selectGetProjection); - this.selectedBaseLayer = this.store.select(mapReducers.selectGetSelectedBaseLayer); - this.panelVisible = this.store.select(mapReducers.selectGetPanelVisible); - this.panelCollapsed = this.store.select(mapReducers.selectGetPanelCollapsed); - this.selectedFeature = this.store.select(mapReducers.selectGetSelectedFeature); - this.selectedItem = this.store.select(mapReducers.selectGetSelectedItem); - this.queryState = this.store.select(mapReducers.selectGetQueryState); - this.clearEnabled = this.store.select(mapReducers.selectGetClearEnabled); - this.searchCollapsed = this.store.select(mapReducers.selectGetSearchCollapsed); - this.searchMinified = this.store.select(mapReducers.selectGetSearchMinified); - this.menuVisible = this.store.select(mapReducers.selectGetMenuVisible); - this.openedModalName = this.store.select(commonReducers.selectOpenedModalName); - this.query = this.store.select(mapReducers.selectGetQuery); - this.extent = this.store.select(mapReducers.selectGetExtent); - this.selectedFeatures.next(null); - this.selectedItemLayer = this.store.select(mapReducers.selectGetSelectedItemLayer); - this.period = this.store.select(mapReducers.selectGetPeriod); - this.position = this.geolocationService.getCurrentPosition(); + this.selectedFeatures$.next({x:0,y:0,features:[]}); + this.mapState$ = this.store.select(mapReducers.selectGetMapState); + this.parentCode$ = this.store.select(mapReducers.selectGetParentCode); + this.features$ = this.store.select(mapReducers.selectGetFeatures); + this.overlayLayers$ = this.store.select(mapReducers.selectGetOverlayLayers); + this.selectedOverlayLayer$ = this.store.select(mapReducers.selectGetSelectedOverlayLayer); + this.baseLayers$ = this.store.select(mapReducers.selectGetBaseLayers); + this.projection$ = this.store.select(mapReducers.selectGetProjection); + this.selectedBaseLayer$ = this.store.select(mapReducers.selectGetSelectedBaseLayer); + this.panelVisible$ = this.store.select(mapReducers.selectGetPanelVisible); + this.panelCollapsed$ = this.store.select(mapReducers.selectGetPanelCollapsed); + this.selectedFeature$ = this.store.select(mapReducers.selectGetSelectedFeature); + this.selectedItem$ = this.store.select(mapReducers.selectGetSelectedItem); + this.queryState$ = this.store.select(mapReducers.selectGetQueryState); + this.clearEnabled$ = this.store.select(mapReducers.selectGetClearEnabled); + this.searchCollapsed$ = this.store.select(mapReducers.selectGetSearchCollapsed); + this.searchMinified$ = this.store.select(mapReducers.selectGetSearchMinified); + this.menuVisible$ = this.store.select(mapReducers.selectGetMenuVisible); + this.openedModalName$ = this.store.select(commonReducers.selectOpenedModalName); + this.query$ = this.store.select(mapReducers.selectGetQuery); + this.extent$ = this.store.select(mapReducers.selectGetExtent); + this.selectedFeatures$.next(null); + this.selectedItemLayer$ = this.store.select(mapReducers.selectGetSelectedItemLayer); + this.period$ = this.store.select(mapReducers.selectGetPeriod); + this.position$ = this.geolocationService.getCurrentPosition(); - this.mapState.pipe(withLatestFrom(this.queryState)).subscribe((state) => { + this.mapState$.pipe(withLatestFrom(this.queryState$)).subscribe((state) => { this.replaceUrl(state[0], state[1], true); }); - this.query.pipe(withLatestFrom(this.mapState)).subscribe((state) => { + this.query$.pipe(withLatestFrom(this.mapState$)).subscribe((state) => { this.replaceUrl(state[1], state[0],false); }); } @@ -194,9 +193,9 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { } this.stateSetCount += 1; }); - setTimeout(() => { - this.map.instance.updateSize(); - }, 500); + // setTimeout(() => { + // this.map.instance.updateSize(); + // }, 500); } handleSearchCollapse(event) { @@ -245,7 +244,6 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { parts.push(mapState.baseLayerCode); parts.push( this.serializeService.serialize(queryState)); this.router.navigate(parts, { replaceUrl: replace,relativeTo:this.route.parent }); - this._ref.markForCheck(); } } @@ -259,7 +257,7 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { let mapState: IMapState = { xCenter: center[0], yCenter: center[1], zoom: zoom, rotation: rotation, baseLayerCode: null }; let state = { mapState: mapState, extent: extent }; let source = from([state]); - source.pipe(withLatestFrom(this.selectedBaseLayer), withLatestFrom(this.queryState)).subscribe(([[state, baselayer], queryState]) => { + source.pipe(withLatestFrom(this.selectedBaseLayer$), withLatestFrom(this.queryState$)).subscribe(([[state, baselayer], queryState]) => { if (mapState && baselayer && queryState) { let newMapState = tassign(state.mapState, { baseLayerCode: baselayer.item.code }); this.replaceUrl(newMapState, tassign(queryState, { bbox: queryState.bboxFilter ? state.extent : queryState.bbox }));