Compare commits
	
		
			110 Commits
		
	
	
		
			392008ae4b
			...
			develop
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 9b86927e2c | ||
|  | e224b80995 | ||
|  | 88f3d9ef1a | ||
| 6737029a61 | |||
|  | d4786564bd | ||
| 6fd69d7999 | |||
|  | 4ec75b3d5a | ||
| 04e3f1e71f | |||
|  | df77631299 | ||
| a8d0f05c81 | |||
|  | c79637be77 | ||
|  | b5e11da9a8 | ||
|  | 9d45c25a95 | ||
|  | 2ffce50c47 | ||
|  | f50ff878e0 | ||
| 09c74448a8 | |||
| 205b73b30e | |||
|  | 313cc59fe7 | ||
| c3fb42c65f | |||
|  | 6b73fb0d55 | ||
| ea641caf38 | |||
| ac15b2018d | |||
|  | fe73f615ad | ||
|  | 3e1b3ee78e | ||
|  | 58391b7828 | ||
|  | b2b2a1e8be | ||
|  | 0d96a1be4f | ||
| d599430cb0 | |||
|  | e17911b339 | ||
|  | 9d6cbffb3d | ||
|  | a11eb81e73 | ||
|  | 4fc3ad933c | ||
|  | e6d90c9e61 | ||
| 250b3841fe | |||
| 51291cd86f | |||
|  | e9e3b719a4 | ||
|  | 1c0ad57b0a | ||
|  | 8684ba25c8 | ||
| 78ca3044cf | |||
| 2e55b5a70e | |||
|  | af0a3dca2d | ||
| f039898c26 | |||
|  | ab19b1c2ce | ||
| a2641f8ea5 | |||
|  | d6841c8f83 | ||
| 187b24ab97 | |||
| b80ddf7854 | |||
|  | 79bc3d9082 | ||
| 1c92ad4c28 | |||
|  | 61e6de0052 | ||
| fb25b92349 | |||
|  | e78f2893ae | ||
| 51170b45e9 | |||
|  | 7365bea145 | ||
| dc096065b4 | |||
|  | b9850311ea | ||
| ee9c050db3 | |||
|  | 95200d9920 | ||
|  | ee1ce21cca | ||
|  | 467f7447bf | ||
|  | 1df4496058 | ||
|  | 2137f3f07e | ||
| a46e8040b2 | |||
|  | 2ca9730735 | ||
| 3d2ca8fb9f | |||
| 6810da8b79 | |||
|  | 4866a6f22d | ||
| 84a1004977 | |||
| 39c619a837 | |||
|  | 758613aa12 | ||
| 6914a61bc4 | |||
| 9f43287433 | |||
|  | 10df983ae3 | ||
| 78249811ab | |||
| df14427d60 | |||
|  | c7c131d437 | ||
| b2fe8f3bd7 | |||
|  | 3027c69fa8 | ||
| 5995d205a4 | |||
|  | fd479c92d8 | ||
| e9880d6eb2 | |||
|  | a4438d8a90 | ||
| 89061774de | |||
|  | 362e6c8534 | ||
| 64a0e58238 | |||
|  | a3d4dfbd59 | ||
| 4c106ebff3 | |||
|  | 5c8fd64521 | ||
| 72ba8eca81 | |||
|  | f89f741265 | ||
| 50af55acb2 | |||
|  | cc879b85e5 | ||
| b4a0a4584d | |||
| 8ab1611f39 | |||
|  | 475c522039 | ||
| 068396ac9a | |||
|  | 4a30707ec6 | ||
| 852b1db895 | |||
|  | fcbd68eca0 | ||
| 3a42f9114e | |||
| 3f664b472c | |||
|  | 57811f3f2e | ||
| 5e1bc6d705 | |||
| d4a230f96d | |||
|  | ed1e1478b2 | ||
| afe013a59a | |||
| db3734f8a3 | |||
|  | 86f75c7f78 | ||
| a7abb4072b | |||
|  | 61d6446d75 | 
							
								
								
									
										54
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										54
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -1,22 +1,22 @@ | |||||||
| { | { | ||||||
|   "name": "farmmaps-lib-app", |   "name": "farmmaps-lib-app", | ||||||
|   "version": "4.12.0", |   "version": "4.19.0", | ||||||
|   "lockfileVersion": 3, |   "lockfileVersion": 3, | ||||||
|   "requires": true, |   "requires": true, | ||||||
|   "packages": { |   "packages": { | ||||||
|     "": { |     "": { | ||||||
|       "name": "farmmaps-lib-app", |       "name": "farmmaps-lib-app", | ||||||
|       "version": "4.12.0", |       "version": "4.19.0", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@angular-eslint/eslint-plugin": "^18.2.0", |         "@angular-eslint/eslint-plugin": "^18.2.0", | ||||||
|         "@angular/animations": "^18.2.3", |         "@angular/animations": "18.2.3", | ||||||
|         "@angular/common": "^18.2.3", |         "@angular/common": "18.2.3", | ||||||
|         "@angular/compiler": "^18.2.3", |         "@angular/compiler": "18.2.3", | ||||||
|         "@angular/core": "^18.2.3", |         "@angular/core": "18.2.3", | ||||||
|         "@angular/forms": "^18.2.3", |         "@angular/forms": "18.2.3", | ||||||
|         "@angular/platform-browser": "^18.2.3", |         "@angular/platform-browser": "18.2.3", | ||||||
|         "@angular/platform-browser-dynamic": "^18.2.3", |         "@angular/platform-browser-dynamic": "18.2.3", | ||||||
|         "@angular/router": "^18.2.3", |         "@angular/router": "18.2.3", | ||||||
|         "@farmmaps/common": "file:dist/common", |         "@farmmaps/common": "file:dist/common", | ||||||
|         "@farmmaps/common-map": "file:dist/common-map", |         "@farmmaps/common-map": "file:dist/common-map", | ||||||
|         "@farmmaps/common-map3d": "file:dist/common-map3d", |         "@farmmaps/common-map3d": "file:dist/common-map3d", | ||||||
| @@ -55,11 +55,11 @@ | |||||||
|       }, |       }, | ||||||
|       "devDependencies": { |       "devDependencies": { | ||||||
|         "@angular-builders/custom-webpack": "^18.0.0", |         "@angular-builders/custom-webpack": "^18.0.0", | ||||||
|         "@angular-devkit/build-angular": "^18.2.3", |         "@angular-devkit/build-angular": "18.2.3", | ||||||
|         "@angular/cli": "^18.2.3", |         "@angular/cli": "18.2.3", | ||||||
|         "@angular/compiler-cli": "^18.2.3", |         "@angular/compiler-cli": "18.2.3", | ||||||
|         "@angular/language-service": "^18.2.3", |         "@angular/language-service": "18.2.3", | ||||||
|         "@angular/localize": "^18.2.3", |         "@angular/localize": "18.2.3", | ||||||
|         "@types/arcgis-rest-api": "^10.4.5", |         "@types/arcgis-rest-api": "^10.4.5", | ||||||
|         "@types/jasmine": "~2.8.8", |         "@types/jasmine": "~2.8.8", | ||||||
|         "@types/jasminewd2": "^2.0.9", |         "@types/jasminewd2": "^2.0.9", | ||||||
| @@ -90,14 +90,14 @@ | |||||||
|     }, |     }, | ||||||
|     "dist/common": { |     "dist/common": { | ||||||
|       "name": "@farmmaps/common", |       "name": "@farmmaps/common", | ||||||
|       "version": "4.12.0-prerelease.2522", |       "version": "4.19.0-prerelease.2582", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "tslib": "^2.3.0" |         "tslib": "^2.3.0" | ||||||
|       }, |       }, | ||||||
|       "peerDependencies": { |       "peerDependencies": { | ||||||
|         "@angular/common": "^18.2.3", |         "@angular/common": "18.2.3", | ||||||
|         "@angular/core": "^18.2.3", |         "@angular/core": "18.2.3", | ||||||
|         "@angular/forms": "^18.2.3", |         "@angular/forms": "18.2.3", | ||||||
|         "@microsoft/signalr": "^3.1.16", |         "@microsoft/signalr": "^3.1.16", | ||||||
|         "@ng-bootstrap/ng-bootstrap": "^17.0.1", |         "@ng-bootstrap/ng-bootstrap": "^17.0.1", | ||||||
|         "@ngrx/effects": "^18.0.2", |         "@ngrx/effects": "^18.0.2", | ||||||
| @@ -115,12 +115,12 @@ | |||||||
|     }, |     }, | ||||||
|     "dist/common-map": { |     "dist/common-map": { | ||||||
|       "name": "@farmmaps/common-map", |       "name": "@farmmaps/common-map", | ||||||
|       "version": "4.12.0-prerelease.2522", |       "version": "4.19.0-prerelease.2582", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "tslib": "^2.0.0" |         "tslib": "^2.0.0" | ||||||
|       }, |       }, | ||||||
|       "peerDependencies": { |       "peerDependencies": { | ||||||
|         "@angular/core": "^18.2.3", |         "@angular/core": "18.2.3", | ||||||
|         "@ngrx/effects": "^18.0.2", |         "@ngrx/effects": "^18.0.2", | ||||||
|         "@ngrx/router-store": "^18.0.2", |         "@ngrx/router-store": "^18.0.2", | ||||||
|         "@ngrx/store": "^18.0.2", |         "@ngrx/store": "^18.0.2", | ||||||
| @@ -130,27 +130,27 @@ | |||||||
|     }, |     }, | ||||||
|     "dist/common-map3d": { |     "dist/common-map3d": { | ||||||
|       "name": "@farmmaps/common-map3d", |       "name": "@farmmaps/common-map3d", | ||||||
|       "version": "4.12.0-prerelease.2522", |       "version": "4.19.0-prerelease.2582", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "tslib": "^2.0.0" |         "tslib": "^2.0.0" | ||||||
|       }, |       }, | ||||||
|       "peerDependencies": { |       "peerDependencies": { | ||||||
|         "@angular/common": "^18.2.3", |         "@angular/common": "18.2.3", | ||||||
|         "@angular/core": "^18.2.3", |         "@angular/core": "18.2.3", | ||||||
|         "cesium": "^1.97.0", |         "cesium": "^1.97.0", | ||||||
|         "ol-cesium": ">=2.13.0" |         "ol-cesium": ">=2.13.0" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "dist/ng-openlayers": { |     "dist/ng-openlayers": { | ||||||
|       "name": "@farmmaps/ng-openlayers", |       "name": "@farmmaps/ng-openlayers", | ||||||
|       "version": "4.12.0-prerelease.2522", |       "version": "4.19.0-prerelease.2582", | ||||||
|       "license": "MPL-2.0", |       "license": "MPL-2.0", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "tslib": "^2.3.0" |         "tslib": "^2.3.0" | ||||||
|       }, |       }, | ||||||
|       "peerDependencies": { |       "peerDependencies": { | ||||||
|         "@angular/common": "^18.2.3", |         "@angular/common": "18.2.3", | ||||||
|         "@angular/core": "^18.2.3", |         "@angular/core": "18.2.3", | ||||||
|         "ol": "^8.2.0" |         "ol": "^8.2.0" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|   | |||||||
							
								
								
									
										28
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "farmmaps-lib-app", |   "name": "farmmaps-lib-app", | ||||||
|   "version": "4.12.0", |   "version": "4.19.0", | ||||||
|   "scripts":   { |   "scripts":   { | ||||||
|     "ng": "ng", |     "ng": "ng", | ||||||
|     "start": "ng serve", |     "start": "ng serve", | ||||||
| @@ -12,14 +12,14 @@ | |||||||
|   "private": true, |   "private": true, | ||||||
|   "dependencies":   { |   "dependencies":   { | ||||||
|     "@angular-eslint/eslint-plugin": "^18.2.0", |     "@angular-eslint/eslint-plugin": "^18.2.0", | ||||||
|     "@angular/animations": "^18.2.3", |     "@angular/animations": "18.2.3", | ||||||
|     "@angular/common": "^18.2.3", |     "@angular/common": "18.2.3", | ||||||
|     "@angular/compiler": "^18.2.3", |     "@angular/compiler": "18.2.3", | ||||||
|     "@angular/core": "^18.2.3", |     "@angular/core": "18.2.3", | ||||||
|     "@angular/forms": "^18.2.3", |     "@angular/forms": "18.2.3", | ||||||
|     "@angular/platform-browser": "^18.2.3", |     "@angular/platform-browser": "18.2.3", | ||||||
|     "@angular/platform-browser-dynamic": "^18.2.3", |     "@angular/platform-browser-dynamic": "18.2.3", | ||||||
|     "@angular/router": "^18.2.3", |     "@angular/router": "18.2.3", | ||||||
|     "@farmmaps/common": "file:dist/common", |     "@farmmaps/common": "file:dist/common", | ||||||
|     "@farmmaps/common-map": "file:dist/common-map", |     "@farmmaps/common-map": "file:dist/common-map", | ||||||
|     "@farmmaps/common-map3d": "file:dist/common-map3d", |     "@farmmaps/common-map3d": "file:dist/common-map3d", | ||||||
| @@ -62,11 +62,11 @@ | |||||||
|   }, |   }, | ||||||
|   "devDependencies":   { |   "devDependencies":   { | ||||||
|     "@angular-builders/custom-webpack": "^18.0.0", |     "@angular-builders/custom-webpack": "^18.0.0", | ||||||
|     "@angular-devkit/build-angular": "^18.2.3", |     "@angular-devkit/build-angular": "18.2.3", | ||||||
|     "@angular/cli": "^18.2.3", |     "@angular/cli": "18.2.3", | ||||||
|     "@angular/compiler-cli": "^18.2.3", |     "@angular/compiler-cli": "18.2.3", | ||||||
|     "@angular/language-service": "^18.2.3", |     "@angular/language-service": "18.2.3", | ||||||
|     "@angular/localize": "^18.2.3", |     "@angular/localize": "18.2.3", | ||||||
|     "@types/arcgis-rest-api": "^10.4.5", |     "@types/arcgis-rest-api": "^10.4.5", | ||||||
|     "@types/jasmine": "~2.8.8", |     "@types/jasmine": "~2.8.8", | ||||||
|     "@types/jasminewd2": "^2.0.9", |     "@types/jasminewd2": "^2.0.9", | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|     "tslib": "^2.0.0" |     "tslib": "^2.0.0" | ||||||
|   }, |   }, | ||||||
|   "peerDependencies": { |   "peerDependencies": { | ||||||
|     "@angular/core": "^18.2.3", |     "@angular/core": "18.2.3", | ||||||
|     "ngrx-store-localstorage": "^18.0.0", |     "ngrx-store-localstorage": "^18.0.0", | ||||||
|     "@ngrx/effects": "^18.0.2", |     "@ngrx/effects": "^18.0.2", | ||||||
|     "@ngrx/router-store": "^18.0.2", |     "@ngrx/router-store": "^18.0.2", | ||||||
|   | |||||||
| @@ -59,6 +59,8 @@ export const TOGGLESHOWDATALAYERSLIDE = '[Map] ToggleShowDataLayerSlide' | |||||||
| export const SETVIEWSTATE = '[Map] SetViewState' | export const SETVIEWSTATE = '[Map] SetViewState' | ||||||
| export const CLEARFEATURES = '[Map] ClearFeatures'; | export const CLEARFEATURES = '[Map] ClearFeatures'; | ||||||
| export const SETPANELEXTRAWIDE = '[Map] SetPanelExtraWide'; | export const SETPANELEXTRAWIDE = '[Map] SetPanelExtraWide'; | ||||||
|  | export const BACKUPFEATURES = '[Map] BackupFeatures'; | ||||||
|  | export const RESTOREFEATURES = '[Map] RestoreFeatures'; | ||||||
|  |  | ||||||
| export class Clear implements Action { | export class Clear implements Action { | ||||||
|   readonly type = CLEAR; |   readonly type = CLEAR; | ||||||
| @@ -347,6 +349,16 @@ export class SetPanelExtraWide implements Action { | |||||||
|   constructor(public panelExtraWide:boolean) {} |   constructor(public panelExtraWide:boolean) {} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | export class BackupFeatures implements Action { | ||||||
|  |   readonly type = BACKUPFEATURES; | ||||||
|  |   constructor() {} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export class RestoreFeatures implements Action { | ||||||
|  |   readonly type = RESTOREFEATURES; | ||||||
|  |   constructor() {} | ||||||
|  | } | ||||||
|  |  | ||||||
| export type Actions = SetMapState | export type Actions = SetMapState | ||||||
|   | Init |   | Init | ||||||
|   | Clear |   | Clear | ||||||
| @@ -395,5 +407,7 @@ export type Actions = SetMapState | |||||||
|   | ToggleShowDataLayerSlide |   | ToggleShowDataLayerSlide | ||||||
|   | SetViewState |   | SetViewState | ||||||
|   | ClearFeatures |   | ClearFeatures | ||||||
|   | SetPanelExtraWide; |   | SetPanelExtraWide | ||||||
|  |   | BackupFeatures | ||||||
|  |   | RestoreFeatures; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -36,6 +36,7 @@ import { RotationResetComponent } from './components/aol/rotation-reset/rotation | |||||||
| import { ZoomToExtentComponent } from './components/aol/zoom-to-extent/zoom-to-extent.component'; | import { ZoomToExtentComponent } from './components/aol/zoom-to-extent/zoom-to-extent.component'; | ||||||
| import { FeatureListContainerComponent } from './components/feature-list-container/feature-list-container.component'; | import { FeatureListContainerComponent } from './components/feature-list-container/feature-list-container.component'; | ||||||
| import { FeatureListCropfieldComponent } from './components/feature-list-cropfield/feature-list-cropfield.component'; | import { FeatureListCropfieldComponent } from './components/feature-list-cropfield/feature-list-cropfield.component'; | ||||||
|  | import { FeatureListObservationComponent } from './components/feature-list-observation/feature-list-observation.component'; | ||||||
| import { FeatureListCroppingschemeComponent } from './components/feature-list-croppingscheme/feature-list-croppingscheme.component'; | import { FeatureListCroppingschemeComponent } from './components/feature-list-croppingscheme/feature-list-croppingscheme.component'; | ||||||
| import { FeatureListFeatureContainerComponent } from './components/feature-list-feature-container/feature-list-feature-container.component'; | import { FeatureListFeatureContainerComponent } from './components/feature-list-feature-container/feature-list-feature-container.component'; | ||||||
| import { FeatureListFeatureCropfieldComponent } from './components/feature-list-feature-cropfield/feature-list-feature-cropfield.component'; | import { FeatureListFeatureCropfieldComponent } from './components/feature-list-feature-cropfield/feature-list-feature-cropfield.component'; | ||||||
| @@ -105,7 +106,7 @@ const metaReducers: Array<MetaReducer<any, any>> = [LocalStorageSync]; | |||||||
|  |  | ||||||
| export { | export { | ||||||
|   AbstractFeatureListComponent, |   AbstractFeatureListComponent, | ||||||
|   AbstractFeatureListFeatureComponent, AbstractItemListComponent, AbstractItemListItemComponent, AbstractItemWidgetComponent, AbstractSelectedItemComponent, DeviceOrientationService, FeatureIconService, FeatureListComponent, FeatureListContainerComponent, FeatureListCropfieldComponent, FeatureListCroppingschemeComponent, FeatureListFeatureComponent, FeatureListFeatureContainerComponent, FeatureListFeatureCropfieldComponent, FeatureListFeatureCroppingschemeComponent, FileDropTargetComponent, ForChild, |   AbstractFeatureListFeatureComponent, AbstractItemListComponent, AbstractItemListItemComponent, AbstractItemWidgetComponent, AbstractSelectedItemComponent, DeviceOrientationService, FeatureIconService, FeatureListComponent, FeatureListContainerComponent, FeatureListCropfieldComponent, FeatureListObservationComponent, FeatureListCroppingschemeComponent, FeatureListFeatureComponent, FeatureListFeatureContainerComponent, FeatureListFeatureCropfieldComponent, FeatureListFeatureCroppingschemeComponent, FileDropTargetComponent, ForChild, | ||||||
|   ForItemType, ForPackage, ForSourceTask, GeolocationService, GeometryThumbnailComponent, GpsLocation, IClickedFeature, ifZoomToShowDirective, IItemLayer, IMapState, IPeriodState, ISelectedFeatures, ItemLayer, ItemLayersComponent, ItemListComponent, ItemListItemComponent, |   ForItemType, ForPackage, ForSourceTask, GeolocationService, GeometryThumbnailComponent, GpsLocation, IClickedFeature, ifZoomToShowDirective, IItemLayer, IMapState, IPeriodState, ISelectedFeatures, ItemLayer, ItemLayersComponent, ItemListComponent, ItemListItemComponent, | ||||||
|   ItemListItemContainerComponent, ITemporalItemLayer, ItemVectorSourceComponent, ItemWidgetListComponent, LayerListComponent, LayerSwitcher, LayerVectorImageComponent, LegendComponent, mapActions, MapComponent, mapEffects, |   ItemListItemContainerComponent, ITemporalItemLayer, ItemVectorSourceComponent, ItemWidgetListComponent, LayerListComponent, LayerSwitcher, LayerVectorImageComponent, LegendComponent, mapActions, MapComponent, mapEffects, | ||||||
|   mapReducers, MapSearchComponent, MetaDataModalComponent, PanToLocation, RotationResetComponent, SelectedItemComponent, SelectedItemContainerComponent, SelectedItemCropfieldComponent, |   mapReducers, MapSearchComponent, MetaDataModalComponent, PanToLocation, RotationResetComponent, SelectedItemComponent, SelectedItemContainerComponent, SelectedItemCropfieldComponent, | ||||||
| @@ -142,6 +143,7 @@ export { | |||||||
|         FeatureListContainerComponent, |         FeatureListContainerComponent, | ||||||
|         FeatureListCroppingschemeComponent, |         FeatureListCroppingschemeComponent, | ||||||
|         FeatureListCropfieldComponent, |         FeatureListCropfieldComponent, | ||||||
|  |         FeatureListObservationComponent, | ||||||
|         FeatureListFeatureContainerComponent, |         FeatureListFeatureContainerComponent, | ||||||
|         FeatureListFeatureComponent, |         FeatureListFeatureComponent, | ||||||
|         FeatureListFeatureCroppingschemeComponent, |         FeatureListFeatureCroppingschemeComponent, | ||||||
| @@ -201,6 +203,7 @@ export { | |||||||
|         FeatureListContainerComponent, |         FeatureListContainerComponent, | ||||||
|         FeatureListCroppingschemeComponent, |         FeatureListCroppingschemeComponent, | ||||||
|         FeatureListCropfieldComponent, |         FeatureListCropfieldComponent, | ||||||
|  |         FeatureListObservationComponent, | ||||||
|         FeatureListFeatureContainerComponent, |         FeatureListFeatureContainerComponent, | ||||||
|         ZoomToExtentComponent, |         ZoomToExtentComponent, | ||||||
|         ifZoomToShowDirective, |         ifZoomToShowDirective, | ||||||
| @@ -214,9 +217,10 @@ export { | |||||||
|         TemporalService, |         TemporalService, | ||||||
|         { provide: AbstractFeatureListComponent, useClass: FeatureListCroppingschemeComponent, multi: true }, |         { provide: AbstractFeatureListComponent, useClass: FeatureListCroppingschemeComponent, multi: true }, | ||||||
|         { provide: AbstractFeatureListComponent, useClass: FeatureListCropfieldComponent, multi: true }, |         { provide: AbstractFeatureListComponent, useClass: FeatureListCropfieldComponent, multi: true }, | ||||||
|  |         { provide: AbstractFeatureListComponent, useClass: FeatureListObservationComponent, multi: true },         | ||||||
|         { provide: AbstractFeatureListFeatureComponent, useClass: FeatureListFeatureComponent, multi: true }, |         { provide: AbstractFeatureListFeatureComponent, useClass: FeatureListFeatureComponent, multi: true }, | ||||||
|         { provide: AbstractFeatureListFeatureComponent, useClass: FeatureListFeatureCroppingschemeComponent, multi: true }, |         { provide: AbstractFeatureListFeatureComponent, useClass: FeatureListFeatureCroppingschemeComponent, multi: true }, | ||||||
|         { provide: AbstractFeatureListFeatureComponent, useClass: FeatureListFeatureCropfieldComponent, multi: true }, |         { provide: AbstractFeatureListFeatureComponent, useClass: FeatureListFeatureCropfieldComponent, multi: true },        | ||||||
|         { provide: AbstractSelectedItemComponent, useClass: SelectedItemComponent, multi: true }, |         { provide: AbstractSelectedItemComponent, useClass: SelectedItemComponent, multi: true }, | ||||||
|         { provide: AbstractSelectedItemComponent, useClass: SelectedItemCropfieldComponent, multi: true }, |         { provide: AbstractSelectedItemComponent, useClass: SelectedItemCropfieldComponent, multi: true }, | ||||||
|         { provide: AbstractSelectedItemComponent, useClass: SelectedItemGeotiffComponent, multi: true }, |         { provide: AbstractSelectedItemComponent, useClass: SelectedItemGeotiffComponent, multi: true }, | ||||||
|   | |||||||
| @@ -0,0 +1,14 @@ | |||||||
|  | <div class="card border-0"> | ||||||
|  |   <div class="card-body" *ngIf="(schemeItem|async);let schemeItem"> | ||||||
|  |     <fm-back-button></fm-back-button> | ||||||
|  |     <h4 i18n>Farm</h4> | ||||||
|  |     <h3>{{schemeItem.name}}</h3> | ||||||
|  |     <div *ngIf="features;let features"> | ||||||
|  |         <div class="cropfields"> | ||||||
|  |           <div class="row m-0 ps-3 pe-3" *ngFor="let feature of features" [ngClass]="{'selected':isFeatureSelected(feature)}" (click)="handleFeatureClick(feature)" (mouseenter)="handleFeatureMouseEnter(feature)" (mouseleave)="handleFeatureMouseLeave(feature)"> | ||||||
|  |           <fm-map-feature-list-feature-container [feature]="feature"></fm-map-feature-list-feature-container> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
| @@ -0,0 +1,20 @@ | |||||||
|  | fm-map-feature-list-feature-container { | ||||||
|  |   width:100%; | ||||||
|  |   pointer-events:none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .row { | ||||||
|  |   border-bottom: 1px solid var(--bs-gray-500); | ||||||
|  |   user-select: none; | ||||||
|  |   padding-left:1.5rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .row.selected { | ||||||
|  |   background-color: var(--bs-gray-100); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .cropfields { | ||||||
|  |   border-top: 1px solid var(--bs-gray-500); | ||||||
|  |   margin-left: -1.25rem; | ||||||
|  |   margin-right: -1.25rem; | ||||||
|  | } | ||||||
| @@ -0,0 +1,30 @@ | |||||||
|  | import { Component, Injectable,AfterViewInit, OnInit,SimpleChanges, ChangeDetectorRef} from '@angular/core'; | ||||||
|  | import { Location } from '@angular/common'; | ||||||
|  | import { AbstractFeatureListComponent } from '../feature-list/feature-list.component'; | ||||||
|  | import {ForItemType } from '../for-item/for-itemtype.decorator'; | ||||||
|  | import {ForChild } from '../for-item/for-child.decorator'; | ||||||
|  | import { Store } from '@ngrx/store'; | ||||||
|  | import * as mapReducers from '../../reducers/map.reducer'; | ||||||
|  | import { commonReducers, ItemTypeService, IItem,ItemService } from '@farmmaps/common'; | ||||||
|  | import { Observable } from 'rxjs'; | ||||||
|  |  | ||||||
|  | @ForChild() | ||||||
|  | @ForItemType("vnd.farmmaps.itemtype.observation") | ||||||
|  | @Injectable() | ||||||
|  | @Component({ | ||||||
|  |   selector: 'fm-map-feature-list-observation', | ||||||
|  |     templateUrl: './feature-list-observation.component.html', | ||||||
|  |     styleUrls: ['./feature-list-observation.component.scss'] | ||||||
|  | }) | ||||||
|  | export class FeatureListObservationComponent extends AbstractFeatureListComponent implements OnInit { | ||||||
|  |  | ||||||
|  |   constructor(store: Store<mapReducers.State | commonReducers.State>, itemTypeService: ItemTypeService, location: Location, private itemService: ItemService) { | ||||||
|  |     super(store, itemTypeService,location); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public schemeItem: Observable<IItem> | ||||||
|  |  | ||||||
|  |   ngOnInit() { | ||||||
|  |     this.schemeItem = this.itemService.getItem(this.queryState.parentCode); | ||||||
|  |   }  | ||||||
|  | } | ||||||
| @@ -10,7 +10,7 @@ | |||||||
|         <li class="nav-item py-0"> |         <li class="nav-item py-0"> | ||||||
|           <span><i class="fal fa-layer-group" aria-hidden="true"></i> <span i18n>Base maps</span></span> |           <span><i class="fal fa-layer-group" aria-hidden="true"></i> <span i18n>Base maps</span></span> | ||||||
|           <div  class="mb-4"> |           <div  class="mb-4"> | ||||||
|             <fm-map-layer-list [baseLayers]="true" [itemLayers]="baseLayers|async" [selectedLayer]="selectedBaseLayer|async" (onSelectLayer)="handleSelectBaseLayer($event)"></fm-map-layer-list> |             <fm-map-layer-list [baseLayers]="true" [itemLayers]="baseMaps|async" [selectedLayer]="selectedBaseLayer|async" (onSelectLayer)="handleSelectBaseLayer($event)"></fm-map-layer-list> | ||||||
|           </div> |           </div> | ||||||
|         </li> |         </li> | ||||||
|         <li class="nav-item py-0"> |         <li class="nav-item py-0"> | ||||||
| @@ -20,12 +20,17 @@ | |||||||
|           </div> |           </div> | ||||||
|         </li> |         </li> | ||||||
|         <li class="nav-item py-0" *ngIf="selectedItemLayer$ | async as selectedItemLayer"> |         <li class="nav-item py-0" *ngIf="selectedItemLayer$ | async as selectedItemLayer"> | ||||||
|           <span><i class="fal fa-layer-group" aria-hidden="true"></i> <span i18n>Data</span><span class="slideButton"><a href="#" title="Compare" class="btn btn-light btn-sm" (click)="handleToggleShowDatalayerSlide($event)"><i class="fal fa-sliders-h-square"></i></a></span></span> |           <span><i class="fal fa-layer-group" aria-hidden="true"></i> <span i18n>Overlay slider</span><span class="slideButton"><a href="#" title="Compare" class="btn btn-light btn-sm" (click)="handleToggleShowDatalayerSlide($event)"><i class="fal fa-sliders-h-square"></i></a></span></span> | ||||||
|           <div class="mb-4"> |           <div class="mb-4"> | ||||||
|             <fm-map-layer-list [dataLayers]="true" [itemLayers]="[selectedItemLayer]" [selectedLayer]="selectedItemLayer" (onToggleVisibility)="handleOnToggleVisibility($event)" (onSetOpacity)="handleOnSetOpacity($event)" (onZoomToExtent)="handleZoomToExtent($event)" (onSelectLayer)="handleSelectOverlayLayer($event)"></fm-map-layer-list> |             <fm-map-layer-list [dataLayers]="true" [itemLayers]="[selectedItemLayer]" [selectedLayer]="selectedItemLayer" (onToggleVisibility)="handleOnToggleVisibility($event)" (onSetOpacity)="handleOnSetOpacity($event)" (onZoomToExtent)="handleZoomToExtent($event)" (onSelectLayer)="handleSelectOverlayLayer($event)"></fm-map-layer-list> | ||||||
|           </div> |           </div> | ||||||
|         </li> |         </li> | ||||||
|  |         <li class="nav-item py-0"> | ||||||
|  |           <span><i class="fal fa-layer-group" aria-hidden="true"></i> <span i18n>Aerial photos</span></span> | ||||||
|  |           <div  class="mb-4"> | ||||||
|  |             <fm-map-layer-list [baseLayers]="true" [itemLayers]="aerialMaps|async" [selectedLayer]="selectedBaseLayer|async" (onSelectLayer)="handleSelectBaseLayer($event)"></fm-map-layer-list> | ||||||
|  |           </div> | ||||||
|  |         </li> | ||||||
|       </ul> |       </ul> | ||||||
|   </div> |   </div> | ||||||
|   </div> |   </div> | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ import { Store } from '@ngrx/store'; | |||||||
| 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 {createEmpty,extend } from 'ol/extent'; | import {createEmpty,extend } from 'ol/extent'; | ||||||
| import { Observable } from 'rxjs'; | import { filter, map, Observable } from 'rxjs'; | ||||||
|  |  | ||||||
| @Component({ | @Component({ | ||||||
|   selector: 'fm-map-layer-switcher', |   selector: 'fm-map-layer-switcher', | ||||||
| @@ -17,7 +17,8 @@ export class LayerSwitcher  implements OnInit,OnChanges{ | |||||||
|   public overlayLayers: Observable<Array<IItemLayer>>; |   public overlayLayers: Observable<Array<IItemLayer>>; | ||||||
|   public selectedOverlayLayer: Observable<IItemLayer>; |   public selectedOverlayLayer: Observable<IItemLayer>; | ||||||
|   public selectedItemLayer$: Observable<IItemLayer>; |   public selectedItemLayer$: Observable<IItemLayer>; | ||||||
|   public baseLayers: Observable<Array<IItemLayer>>; |   public baseMaps: Observable<Array<IItemLayer>>; | ||||||
|  |   public aerialMaps: Observable<Array<IItemLayer>>; | ||||||
|   public selectedBaseLayer: Observable<IItemLayer>; |   public selectedBaseLayer: Observable<IItemLayer>; | ||||||
|  |  | ||||||
|   constructor( private store: Store<mapReducers.State>) { |   constructor( private store: Store<mapReducers.State>) { | ||||||
| @@ -26,7 +27,8 @@ export class LayerSwitcher  implements OnInit,OnChanges{ | |||||||
|   ngOnInit() {  |   ngOnInit() {  | ||||||
|     this.overlayLayers = this.store.select(mapReducers.selectGetOverlayLayers); |     this.overlayLayers = this.store.select(mapReducers.selectGetOverlayLayers); | ||||||
|     this.selectedOverlayLayer = this.store.select(mapReducers.selectGetSelectedOverlayLayer); |     this.selectedOverlayLayer = this.store.select(mapReducers.selectGetSelectedOverlayLayer); | ||||||
|     this.baseLayers = this.store.select(mapReducers.selectGetBaseLayers); |     this.baseMaps = this.store.select(mapReducers.selectGetBaseMaps); | ||||||
|  |     this.aerialMaps = this.store.select(mapReducers.selectGetArealMaps); | ||||||
|     this.selectedBaseLayer = this.store.select(mapReducers.selectGetSelectedBaseLayer); |     this.selectedBaseLayer = this.store.select(mapReducers.selectGetSelectedBaseLayer); | ||||||
|     this.selectedItemLayer$ = this.store.select(mapReducers.selectGetSelectedItemLayer) |     this.selectedItemLayer$ = this.store.select(mapReducers.selectGetSelectedItemLayer) | ||||||
|     this.showLayerSwitcher = this.store.select(mapReducers.selectGetShowLayerSwitcher); |     this.showLayerSwitcher = this.store.select(mapReducers.selectGetShowLayerSwitcher); | ||||||
| @@ -83,3 +85,4 @@ export class LayerSwitcher  implements OnInit,OnChanges{ | |||||||
|     event.preventDefault(); |     event.preventDefault(); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |   | ||||||
| @@ -52,11 +52,20 @@ aol-map { position:absolute;width:100%;height:calc( 100vh );} | |||||||
|   pointer-events: none; |   pointer-events: none; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @media screen and (max-width: 768px) { | ||||||
|  |   .control-container { | ||||||
|  |     margin-left: 1em; | ||||||
|  |     margin-right: 1em; | ||||||
|  |     pointer-events: all; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
| .control-container { | @media screen and (min-width: 768px) { | ||||||
|   float:right; |   .control-container {     | ||||||
|   margin-right: 1em; |     float: right; | ||||||
|   pointer-events: all; |     margin-right: 1em; | ||||||
|  |     pointer-events: all; | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| .fullscreen .viewport-container { | .fullscreen .viewport-container { | ||||||
|   | |||||||
| @@ -1,39 +1,39 @@ | |||||||
| import { Component, OnInit, OnDestroy, HostListener, ViewChild, AfterViewInit,NgZone,ElementRef } from '@angular/core'; | import { Component, OnInit, OnDestroy, HostListener, ViewChild, AfterViewInit, NgZone, ElementRef } from '@angular/core'; | ||||||
| import { Location } from '@angular/common'; | import { Location } from '@angular/common'; | ||||||
| import { Observable, Subject, Subscription, from,of ,EMPTY } from 'rxjs'; | import { Observable, Subject, Subscription, from, of, EMPTY } from 'rxjs'; | ||||||
| import { withLatestFrom, switchMap,skip  } from 'rxjs/operators'; | import { withLatestFrom, switchMap, skip } from 'rxjs/operators'; | ||||||
| import { Router, ActivatedRoute, ParamMap } from '@angular/router'; | import { Router, ActivatedRoute, ParamMap } from '@angular/router'; | ||||||
| import { Store } from '@ngrx/store'; | import { Store } from '@ngrx/store'; | ||||||
| import { DeviceService } from '@farmmaps/common'; | import { DeviceService } from '@farmmaps/common'; | ||||||
| import {getRenderPixel} from 'ol/render'; | import { getRenderPixel } from 'ol/render'; | ||||||
|  |  | ||||||
| // Map | // Map | ||||||
| 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 { 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'; | ||||||
| import { IListItem, IQueryState } from '@farmmaps/common'; | import { IListItem, IQueryState } from '@farmmaps/common'; | ||||||
| import { IPeriodState } from '../../models/period.state'; | import { IPeriodState } from '../../models/period.state'; | ||||||
| import {IStyles} from '../../models/style.cache'; | import { IStyles } from '../../models/style.cache'; | ||||||
| import { IDroppedFile } from '../aol/file-drop-target/file-drop-target.component'; | import { IDroppedFile } from '../aol/file-drop-target/file-drop-target.component'; | ||||||
| import { StateSerializerService } from '@farmmaps/common'; | import { StateSerializerService } from '@farmmaps/common'; | ||||||
| import { GeolocationService} from '../../services/geolocation.service'; | import { GeolocationService } from '../../services/geolocation.service'; | ||||||
| import { GeolocatorService } from '@farmmaps/common'; | import { GeolocatorService } from '@farmmaps/common'; | ||||||
| import {DeviceOrientationService} from '../../services/device-orientation.service'; | import { DeviceOrientationService } from '../../services/device-orientation.service'; | ||||||
|  |  | ||||||
| // AppCommon | // AppCommon | ||||||
| import { ResumableFileUploadService, ItemTypeService } from '@farmmaps/common'; | import { ResumableFileUploadService, ItemTypeService } from '@farmmaps/common'; | ||||||
| import { IItemType, IItem } from '@farmmaps/common'; | import { IItemType, IItem } from '@farmmaps/common'; | ||||||
| import {commonReducers} from '@farmmaps/common'; | import { commonReducers } from '@farmmaps/common'; | ||||||
| import {commonActions} from '@farmmaps/common'; | import { commonActions } from '@farmmaps/common'; | ||||||
|  |  | ||||||
| import {Feature} from 'ol'; | import { Feature } from 'ol'; | ||||||
| import {Geometry,Point,Circle} from 'ol/geom'; | import { Geometry, Point, Circle } from 'ol/geom'; | ||||||
| import {Extent,createEmpty,extend } from 'ol/extent'; | import { Extent, createEmpty, extend } from 'ol/extent'; | ||||||
| import {transform} from 'ol/proj'; | import { transform } from 'ol/proj'; | ||||||
| import { tassign } from 'tassign'; | import { tassign } from 'tassign'; | ||||||
| import * as style from 'ol/style'; | import * as style from 'ol/style'; | ||||||
|  |  | ||||||
| @@ -44,7 +44,7 @@ import * as style from 'ol/style'; | |||||||
|   styleUrls: ['./map.component.scss'] |   styleUrls: ['./map.component.scss'] | ||||||
| }) | }) | ||||||
|  |  | ||||||
| export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | export class MapComponent implements OnInit, OnDestroy, AfterViewInit { | ||||||
|   title = 'Map'; |   title = 'Map'; | ||||||
|   public openedModalName$: Observable<string> = this.store.select(commonReducers.selectOpenedModalName); |   public openedModalName$: Observable<string> = this.store.select(commonReducers.selectOpenedModalName); | ||||||
|   public itemTypes$: Observable<{ [id: string]: IItemType }>; |   public itemTypes$: Observable<{ [id: string]: IItemType }>; | ||||||
| @@ -54,49 +54,51 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | |||||||
|   public selectedOverlayLayer$: Observable<IItemLayer> = this.store.select(mapReducers.selectGetSelectedOverlayLayer); |   public selectedOverlayLayer$: Observable<IItemLayer> = this.store.select(mapReducers.selectGetSelectedOverlayLayer); | ||||||
|   public selectedItemLayer$: Observable<IItemLayer> = this.store.select(mapReducers.selectGetSelectedItemLayer); |   public selectedItemLayer$: Observable<IItemLayer> = this.store.select(mapReducers.selectGetSelectedItemLayer); | ||||||
|   public baseLayers$: Observable<Array<IItemLayer>> = this.store.select(mapReducers.selectGetBaseLayers); |   public baseLayers$: Observable<Array<IItemLayer>> = this.store.select(mapReducers.selectGetBaseLayers); | ||||||
|   public selectedBaseLayer$: Observable<IItemLayer> =  this.store.select(mapReducers.selectGetSelectedBaseLayer); |   public selectedBaseLayer$: Observable<IItemLayer> = this.store.select(mapReducers.selectGetSelectedBaseLayer); | ||||||
|   public projection$: Observable<string> = this.store.select(mapReducers.selectGetProjection); |   public projection$: Observable<string> = this.store.select(mapReducers.selectGetProjection); | ||||||
|   public selectedFeatures$: Subject<ISelectedFeatures> = new Subject<ISelectedFeatures>(); |   public selectedFeatures$: Subject<ISelectedFeatures> = new Subject<ISelectedFeatures>(); | ||||||
|   public droppedFile$: Subject<IDroppedFile>  = new Subject<IDroppedFile>(); |   public droppedFile$: Subject<IDroppedFile> = new Subject<IDroppedFile>(); | ||||||
|   private paramSub: Subscription; |   private paramSub: Subscription; | ||||||
|   private itemTypeSub: Subscription; |   private itemTypeSub: Subscription; | ||||||
|   private stateSub: Subscription; |   private stateSub: Subscription; | ||||||
|   private queryStateSub: Subscription; |   private queryStateSub: Subscription; | ||||||
|   private querySub: Subscription; |   private querySub: Subscription; | ||||||
|   public parentCode$: Observable<string> =this.store.select(mapReducers.selectGetParentCode); |   public parentCode$: Observable<string> = this.store.select(mapReducers.selectGetParentCode); | ||||||
|   public panelVisible$: Observable<boolean> = this.store.select(mapReducers.selectGetPanelVisible); |   public panelVisible$: Observable<boolean> = this.store.select(mapReducers.selectGetPanelVisible); | ||||||
|   public panelCollapsed$: Observable<boolean> = this.store.select(mapReducers.selectGetPanelCollapsed); |   public panelCollapsed$: Observable<boolean> = this.store.select(mapReducers.selectGetPanelCollapsed); | ||||||
|   public panelExtraWide$: Observable<boolean> = this.store.select(mapReducers.selectGetPanelExtraWide); |   public panelExtraWide$: Observable<boolean> = this.store.select(mapReducers.selectGetPanelExtraWide); | ||||||
|   public selectedFeature$: Observable<Feature<Geometry>> = this.store.select(mapReducers.selectGetSelectedFeature); |   public selectedFeature$: Observable<Feature<Geometry>> = this.store.select(mapReducers.selectGetSelectedFeature); | ||||||
|   public clickedFeature: Subject<Feature<Geometry>> = new Subject<Feature<Geometry>>(); |   public clickedFeature: Subject<Feature<Geometry>> = new Subject<Feature<Geometry>>(); | ||||||
|   public selectedItem$: Observable<IItem> = this.store.select(mapReducers.selectGetSelectedItem); |   public selectedItem$: Observable<IItem> = this.store.select(mapReducers.selectGetSelectedItem); | ||||||
|   public parentItem$: Observable<IItem> =this.store.select(mapReducers.selectGetParentItem); |   public parentItem$: Observable<IItem> = this.store.select(mapReducers.selectGetParentItem); | ||||||
|   public queryState$: Observable<IQueryState> = this.store.select(mapReducers.selectGetQueryState); |   public queryState$: Observable<IQueryState> = this.store.select(mapReducers.selectGetQueryState); | ||||||
|   public state$:Observable<{mapState:IMapState,queryState:IQueryState}> = this.store.select(mapReducers.selectGetState); |   public state$: Observable<{ mapState: IMapState, queryState: IQueryState }> = this.store.select(mapReducers.selectGetState); | ||||||
|   public period$: Observable<IPeriodState> = this.store.select(mapReducers.selectGetPeriod); |   public period$: Observable<IPeriodState> = this.store.select(mapReducers.selectGetPeriod); | ||||||
|   public clearEnabled$: Observable<boolean> = this.store.select(mapReducers.selectGetClearEnabled); |   public clearEnabled$: Observable<boolean> = this.store.select(mapReducers.selectGetClearEnabled); | ||||||
|   public searchCollapsed$: Observable<boolean> = this.store.select(mapReducers.selectGetSearchCollapsed); |   public searchCollapsed$: Observable<boolean> = this.store.select(mapReducers.selectGetSearchCollapsed); | ||||||
|   public searchMinified$: Observable<boolean> = this.store.select(mapReducers.selectGetSearchMinified); |   public searchMinified$: Observable<boolean> = this.store.select(mapReducers.selectGetSearchMinified); | ||||||
|   public showDataLayerSlide$: Observable<boolean> = this.store.select(mapReducers.selectGetShowdataLayerSlide); |   public showDataLayerSlide$: Observable<boolean> = this.store.select(mapReducers.selectGetShowdataLayerSlide); | ||||||
|   public menuVisible$: Observable<boolean>; |   public menuVisible$: Observable<boolean>; | ||||||
|   public query$: Observable<IQuery> =  this.store.select(mapReducers.selectGetQuery); |   public query$: Observable<IQuery> = this.store.select(mapReducers.selectGetQuery); | ||||||
|   public position$: Observable<GeolocationPosition> = this.geolocationService.getCurrentPosition(); |   public position$: Observable<GeolocationPosition> = this.geolocationService.getCurrentPosition(); | ||||||
|   public compassHeading$: Observable<number> = this.deviceorientationService.getCurrentCompassHeading(); |   public compassHeading$: Observable<number> = this.deviceorientationService.getCurrentCompassHeading(); | ||||||
|   public baseLayersCollapsed = true; |   public baseLayersCollapsed = true; | ||||||
|   public overlayLayersCollapsed = true; |   public overlayLayersCollapsed = true; | ||||||
|   public extent$: Observable<Extent> = this.store.select(mapReducers.selectGetExtent); |   public extent$: Observable<Extent> = this.store.select(mapReducers.selectGetExtent); | ||||||
|   public styles$:Observable<IStyles> = this.store.select(mapReducers.selectGetStyles); |   public styles$: Observable<IStyles> = this.store.select(mapReducers.selectGetStyles); | ||||||
|   public fullscreen$: Observable<boolean> = this.store.select(commonReducers.selectGetFullScreen); |   public fullscreen$: Observable<boolean> = this.store.select(commonReducers.selectGetFullScreen); | ||||||
|   private lastUrl = ""; |   private lastUrl = ""; | ||||||
|   private initialized = false; |   private initialized = false; | ||||||
|   public noContent = false; |   public noContent = false; | ||||||
|   public overrideSelectedItemLayer = false; |   public overrideSelectedItemLayer = false; | ||||||
|   public overrideOverlayLayers = false; |   public overrideOverlayLayers = false; | ||||||
|  |   public hideShowLayerValues = false; | ||||||
|  |   public const | ||||||
|   public dataLayerSlideValue = 50; |   public dataLayerSlideValue = 50; | ||||||
|   public dataLayerSlideEnabled = false; |   public dataLayerSlideEnabled = false; | ||||||
|   private visibleAreaBottom = 0; |   private visibleAreaBottom = 0; | ||||||
|   private viewEnabled = true; |   private viewEnabled = true; | ||||||
|    |  | ||||||
|   @ViewChild('map') map; |   @ViewChild('map') map; | ||||||
|   @ViewChild('contentDiv') contentDiv: ElementRef; |   @ViewChild('contentDiv') contentDiv: ElementRef; | ||||||
|  |  | ||||||
| @@ -110,47 +112,48 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | |||||||
|     private geolocationService: GeolocationService, |     private geolocationService: GeolocationService, | ||||||
|     private geolocaterService: GeolocatorService, |     private geolocaterService: GeolocatorService, | ||||||
|     private zone: NgZone, |     private zone: NgZone, | ||||||
|     private deviceorientationService:DeviceOrientationService, |     private deviceorientationService: DeviceOrientationService, | ||||||
|     public devicesService:DeviceService) { |     public devicesService: DeviceService) { | ||||||
|       if(route && route.snapshot && route.snapshot.data && route.snapshot.data["fm-map-map"]) { |     if (route && route.snapshot && route.snapshot.data && route.snapshot.data["fm-map-map"]) { | ||||||
|         const params = route.snapshot.data["fm-map-map"]; |       const params = route.snapshot.data["fm-map-map"]; | ||||||
|         this.overrideSelectedItemLayer = params["overrideSelectedItemlayer"] ? params["overrideSelectedItemlayer"] : false; |       this.overrideSelectedItemLayer = params["overrideSelectedItemlayer"] ? params["overrideSelectedItemlayer"] : false; | ||||||
|         this.overrideOverlayLayers = params["overrideOverlayLayers"] ? params["overrideOverlayLayers"] : false; |       this.overrideOverlayLayers = params["overrideOverlayLayers"] ? params["overrideOverlayLayers"] : false; | ||||||
|       } |       this.hideShowLayerValues = params["hideShowLayerValues"] ? params["hideShowLayerValues"] : false; | ||||||
|       this.querySub = this.query$.pipe(skip(1), withLatestFrom(this.mapState$)).subscribe(([query,mapState]) =>{ |     } | ||||||
|         if(query && query.querystate) { |     this.querySub = this.query$.pipe(skip(1), withLatestFrom(this.mapState$)).subscribe(([query, mapState]) => { | ||||||
|           let newQueryState = tassign(mapReducers.initialQueryState); |       if (query && query.querystate) { | ||||||
|           //console.debug(`Do Query`); |         let newQueryState = tassign(mapReducers.initialQueryState); | ||||||
|           const urlparts=[]; |         //console.debug(`Do Query`); | ||||||
|           if (query.querystate.itemCode && query.querystate.itemCode != "") { |         const urlparts = []; | ||||||
|             if(query.querystate.itemType && query.querystate.itemType!= "") { |         if (query.querystate.itemCode && query.querystate.itemCode != "") { | ||||||
|               const itemType = this.itemTypeService.itemTypes[query.querystate.itemType]; |           if (query.querystate.itemType && query.querystate.itemType != "") { | ||||||
|               if (itemType && itemType.viewer && itemType.viewer == "edit_in_editor" && itemType.editor) { |             const itemType = this.itemTypeService.itemTypes[query.querystate.itemType]; | ||||||
|                 urlparts.push('/editor'); |             if (itemType && itemType.viewer && itemType.viewer == "edit_in_editor" && itemType.editor) { | ||||||
|                 urlparts.push(itemType.editor); |               urlparts.push('/editor'); | ||||||
|                 urlparts.push('item'); |               urlparts.push(itemType.editor); | ||||||
|                 urlparts.push(query.querystate.itemCode); |               urlparts.push('item'); | ||||||
|               } |               urlparts.push(query.querystate.itemCode); | ||||||
|             } |             } | ||||||
|           } else { |  | ||||||
|             newQueryState= query.querystate; |  | ||||||
|           } |  | ||||||
|           if(urlparts.length==0 ) { |  | ||||||
|             newQueryState.itemCode = query.querystate.itemCode; |  | ||||||
|             this.zone.run(() => { |  | ||||||
|               this.replaceUrl(mapState,newQueryState,query.replace);              |  | ||||||
|             }) |  | ||||||
|           } else { |  | ||||||
|             this.router.navigate(urlparts); |  | ||||||
|           } |           } | ||||||
|  |         } else { | ||||||
|  |           newQueryState = query.querystate; | ||||||
|         } |         } | ||||||
|       }); |         if (urlparts.length == 0) { | ||||||
|       this.store.dispatch(new mapActions.Init()); |           newQueryState.itemCode = query.querystate.itemCode; | ||||||
|       // this.store.select(commonReducers.getRootItems).subscribe((l) => { |           this.zone.run(() => { | ||||||
|       //    if(l && l.length>0) { |             this.replaceUrl(mapState, newQueryState, query.replace); | ||||||
|       //      this.store.dispatch(new mapActions.Init()); |           }) | ||||||
|       //    } |         } else { | ||||||
|       // }); |           this.router.navigate(urlparts); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |     this.store.dispatch(new mapActions.Init()); | ||||||
|  |     // this.store.select(commonReducers.getRootItems).subscribe((l) => { | ||||||
|  |     //    if(l && l.length>0) { | ||||||
|  |     //      this.store.dispatch(new mapActions.Init()); | ||||||
|  |     //    } | ||||||
|  |     // }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @HostListener('document:keyup', ['$event']) |   @HostListener('document:keyup', ['$event']) | ||||||
| @@ -161,27 +164,27 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handlePanelResize(resizeTop:number) { |   handlePanelResize(resizeTop: number) { | ||||||
|     if(resizeTop==100 || !this.devicesService.IsMobile() ) { |     if (resizeTop == 100 || !this.devicesService.IsMobile()) { | ||||||
|       this.visibleAreaBottom=0; |       this.visibleAreaBottom = 0; | ||||||
|     } else { |     } else { | ||||||
|       this.visibleAreaBottom=100-resizeTop; |       this.visibleAreaBottom = 100 - resizeTop; | ||||||
|       if(this.visibleAreaBottom>60) { |       if (this.visibleAreaBottom > 60) { | ||||||
|         this.visibleAreaBottom=60; |         this.visibleAreaBottom = 60; | ||||||
|       } |       } | ||||||
|     }       |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   bottom(panelVisible:boolean) { |   bottom(panelVisible: boolean) { | ||||||
|     if(panelVisible) { |     if (panelVisible) { | ||||||
|       return this.visibleAreaBottom + '%';      |       return this.visibleAreaBottom + '%'; | ||||||
|     } else { |     } else { | ||||||
|       return "0%"; |       return "0%"; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleOpenModal(modalName: string) { |   handleOpenModal(modalName: string) { | ||||||
|     this.store.dispatch(new  commonActions.OpenModal(modalName)); |     this.store.dispatch(new commonActions.OpenModal(modalName)); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleCloseModal() { |   handleCloseModal() { | ||||||
| @@ -189,7 +192,7 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleFileDropped(droppedFile: IDroppedFile) { |   handleFileDropped(droppedFile: IDroppedFile) { | ||||||
|     this.uploadService.addFiles(droppedFile.files, droppedFile.event, { parentCode:droppedFile.parentCode, geometry:droppedFile.geometry }); |     this.uploadService.addFiles(droppedFile.files, droppedFile.event, { parentCode: droppedFile.parentCode, geometry: droppedFile.geometry }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleFeatureClick(feature: Feature<Geometry>) { |   handleFeatureClick(feature: Feature<Geometry>) { | ||||||
| @@ -205,26 +208,26 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | |||||||
|     this.store.dispatch(new mapActions.DoQuery(queryState)); |     this.store.dispatch(new mapActions.DoQuery(queryState)); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleSidepaneloutletActivate(component:any) { |   handleSidepaneloutletActivate(component: any) { | ||||||
|    if(component && component.hasOwnProperty('clickedFeature')) { |     if (component && component.hasOwnProperty('clickedFeature')) { | ||||||
|      (component as IClickedFeature).clickedFeature = this.clickedFeature; |       (component as IClickedFeature).clickedFeature = this.clickedFeature; | ||||||
|    } |     } | ||||||
|    if(component && component.hasOwnProperty('extrawide')) { |     if (component && component.hasOwnProperty('extrawide')) { | ||||||
|     this.store.dispatch(new mapActions.SetPanelExtraWide(true)); |       this.store.dispatch(new mapActions.SetPanelExtraWide(true)); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|  } |  | ||||||
|  |  | ||||||
|   handleSidepaneloutletDeactivate(component:any) { |   handleSidepaneloutletDeactivate(component: any) { | ||||||
|     if(component && component.hasOwnProperty('clickedFeature')) { |     if (component && component.hasOwnProperty('clickedFeature')) { | ||||||
|       (component as IClickedFeature).clickedFeature = null; |       (component as IClickedFeature).clickedFeature = null; | ||||||
|     } |     } | ||||||
|     if(component && component.hasOwnProperty('extrawide')) { |     if (component && component.hasOwnProperty('extrawide')) { | ||||||
|       this.store.dispatch(new mapActions.SetPanelExtraWide(false)); |       this.store.dispatch(new mapActions.SetPanelExtraWide(false)); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handlePrerender(event:any) { |   handlePrerender(event: any) { | ||||||
|     if(!this.dataLayerSlideEnabled) return; |     if (!this.dataLayerSlideEnabled) return; | ||||||
|     const ctx = event.context; |     const ctx = event.context; | ||||||
|     const mapSize = this.map.instance.getSize(); |     const mapSize = this.map.instance.getSize(); | ||||||
|     const width = mapSize[0] * (this.dataLayerSlideValue / 100); |     const width = mapSize[0] * (this.dataLayerSlideValue / 100); | ||||||
| @@ -243,72 +246,74 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | |||||||
|     ctx.clip(); |     ctx.clip(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleSlideChange(event:any) { |   handleSlideChange(event: any) { | ||||||
|      this.dataLayerSlideValue = event.target.value; |     this.dataLayerSlideValue = event.target.value; | ||||||
|      this.map.instance.render(); |     this.map.instance.render(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   ngOnInit() {     |   ngOnInit() { | ||||||
|     this.initialized = false; |     this.initialized = false; | ||||||
|     //console.debug("Init"); |     //console.debug("Init"); | ||||||
|     this.store.dispatch(new mapActions.Clear()); |     this.store.dispatch(new mapActions.Clear()); | ||||||
|     this.selectedFeatures$.next({x:0,y:0,features:[]}); |     this.selectedFeatures$.next({ x: 0, y: 0, features: [] }); | ||||||
|     this.selectedFeatures$.next(null); |     this.selectedFeatures$.next(null); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   initCustomStyles() { |   initCustomStyles() { | ||||||
|     this.store.dispatch(new mapActions.SetStyle('vnd.farmmaps.itemtype.layer',new style.Style({ |     this.store.dispatch(new mapActions.SetStyle('vnd.farmmaps.itemtype.layer', new style.Style({ | ||||||
|       stroke: new style.Stroke({ |       stroke: new style.Stroke({ | ||||||
|         color: 'red', |         color: 'red', | ||||||
|         lineDash: [ 5,5], |         lineDash: [5, 5], | ||||||
|         width: 1 |         width: 1 | ||||||
|       }), |       }), | ||||||
|       geometry:(feature) =>feature.getGeometry() |       geometry: (feature) => feature.getGeometry() | ||||||
|     }))); |     }))); | ||||||
|     this.store.dispatch(new mapActions.SetStyle('vnd.farmmaps.itemtype.layer_selected',new style.Style({ |     this.store.dispatch(new mapActions.SetStyle('vnd.farmmaps.itemtype.layer_selected', new style.Style({ | ||||||
|       stroke: new style.Stroke({ |       stroke: new style.Stroke({ | ||||||
|         color: 'red', |         color: 'red', | ||||||
|         lineDash: [ 5,5], |         lineDash: [5, 5], | ||||||
|         width: 3 |         width: 3 | ||||||
|       }), |       }), | ||||||
|       geometry:(feature) =>feature.getGeometry() |       geometry: (feature) => feature.getGeometry() | ||||||
|     }))); |     }))); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   round(value:number,decimals:number):number { |   round(value: number, decimals: number): number { | ||||||
|     const d = Math.pow(10, decimals); |     const d = Math.pow(10, decimals); | ||||||
|     return Math.round((value + Number.EPSILON)*d)/d; |     return Math.round((value + Number.EPSILON) * d) / d; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   getMapStateFromUrl(params:ParamMap):IMapState { |   getMapStateFromUrl(params: ParamMap): IMapState { | ||||||
|     const hasUrlmapState = params.has("xCenter") && params.has("yCenter"); |     const hasUrlmapState = params.has("xCenter") && params.has("yCenter"); | ||||||
|     if (hasUrlmapState) { |     if (hasUrlmapState) { | ||||||
|       const xCenter = parseFloat(params.get("xCenter")); |       const xCenter = parseFloat(params.get("xCenter")); | ||||||
|       const yCenter = parseFloat(params.get("yCenter")); |       const yCenter = parseFloat(params.get("yCenter")); | ||||||
|       const zoom = parseFloat(params.get("zoom")); |       const zoom = parseFloat(params.get("zoom")); | ||||||
|       const rotation = parseFloat(params.get("rotation")); |       const rotation = parseFloat(params.get("rotation")); | ||||||
|       const baseLayer = params.get("baseLayer")?params.get("baseLayer"):""; |       const baseLayer = params.get("baseLayer") ? params.get("baseLayer") : ""; | ||||||
|       const newMapState = {zoom: zoom, rotation: rotation, xCenter: xCenter, yCenter: yCenter, baseLayerCode: baseLayer }; |       const newMapState = { zoom: zoom, rotation: rotation, xCenter: xCenter, yCenter: yCenter, baseLayerCode: baseLayer }; | ||||||
|       return newMapState; |       return newMapState; | ||||||
|     } else { |     } else { | ||||||
|       return null; |       return null; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   normalizeMapState(mapState:IMapState):IMapState { |   normalizeMapState(mapState: IMapState): IMapState { | ||||||
|     if(!mapState) return null; |     if (!mapState) return null; | ||||||
|     return {zoom: this.round(mapState.zoom,0), |     return { | ||||||
|       rotation: this.round(mapState.rotation,2), |       zoom: this.round(mapState.zoom, 0), | ||||||
|       xCenter: this.round(mapState.xCenter,5), |       rotation: this.round(mapState.rotation, 2), | ||||||
|       yCenter: this.round(mapState.yCenter,5), |       xCenter: this.round(mapState.xCenter, 5), | ||||||
|       baseLayerCode: mapState.baseLayerCode }; |       yCenter: this.round(mapState.yCenter, 5), | ||||||
|  |       baseLayerCode: mapState.baseLayerCode | ||||||
|  |     }; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   serializeMapState(mapState:IMapState):string { |   serializeMapState(mapState: IMapState): string { | ||||||
|     return JSON.stringify(this.normalizeMapState(mapState)); |     return JSON.stringify(this.normalizeMapState(mapState)); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   getQueryStateFromUrl(params:ParamMap):IQueryState { |   getQueryStateFromUrl(params: ParamMap): IQueryState { | ||||||
|     if (params.has("queryState")) { |     if (params.has("queryState")) { | ||||||
|       const queryState = params.get("queryState"); |       const queryState = params.get("queryState"); | ||||||
|       let newQueryState = tassign(mapReducers.initialQueryState); |       let newQueryState = tassign(mapReducers.initialQueryState); | ||||||
| @@ -323,65 +328,65 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | |||||||
|  |  | ||||||
|   ngAfterViewInit() { |   ngAfterViewInit() { | ||||||
|     //console.debug("View init"); |     //console.debug("View init"); | ||||||
|     this.noContent=true; |     this.noContent = true; | ||||||
|     this.route.children.forEach((entry) => { |     this.route.children.forEach((entry) => { | ||||||
|       if(entry.outlet=="primary") { |       if (entry.outlet == "primary") { | ||||||
|         this.noContent=false; |         this.noContent = false; | ||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|        |  | ||||||
|     this.initCustomStyles(); |     this.initCustomStyles(); | ||||||
|  |  | ||||||
|     // url to state |     // url to state | ||||||
|  |  | ||||||
|       const urlMapState = this.getMapStateFromUrl(this.route.snapshot.paramMap); |     const urlMapState = this.getMapStateFromUrl(this.route.snapshot.paramMap); | ||||||
|       const urlQueryState = this.getQueryStateFromUrl(this.route.snapshot.paramMap); |     const urlQueryState = this.getQueryStateFromUrl(this.route.snapshot.paramMap); | ||||||
|       if(urlQueryState && urlMapState && this.noContent) { |     if (urlQueryState && urlMapState && this.noContent) { | ||||||
|         this.store.dispatch(new mapActions.SetState(urlMapState,urlQueryState)); |       this.store.dispatch(new mapActions.SetState(urlMapState, urlQueryState)); | ||||||
|         window.localStorage.setItem("FarmMapsCommonMap_mapState",this.serializeMapState(urlMapState)); |       window.localStorage.setItem("FarmMapsCommonMap_mapState", this.serializeMapState(urlMapState)); | ||||||
|       } else if(urlQueryState && this.noContent) { |     } else if (urlQueryState && this.noContent) { | ||||||
|         this.store.dispatch(new mapActions.SetQueryState(urlQueryState)); |       this.store.dispatch(new mapActions.SetQueryState(urlQueryState)); | ||||||
|       }  else { |     } else { | ||||||
|         this.store.dispatch(new mapActions.SetReplaceUrl(true)); |       this.store.dispatch(new mapActions.SetReplaceUrl(true)); | ||||||
|       } |     } | ||||||
|  |  | ||||||
|       this.paramSub = this.route.paramMap.pipe(withLatestFrom(this.state$),switchMap(([params,state]) => { |     this.paramSub = this.route.paramMap.pipe(withLatestFrom(this.state$), switchMap(([params, state]) => { | ||||||
|          if(this.initialized && this.noContent) { |       if (this.initialized && this.noContent) { | ||||||
|           const urlQueryState = this.getQueryStateFromUrl(params); |         const urlQueryState = this.getQueryStateFromUrl(params); | ||||||
|           if( this.serializeService.serialize(state.queryState) != this.serializeService.serialize(urlQueryState)) { |         if (this.serializeService.serialize(state.queryState) != this.serializeService.serialize(urlQueryState)) { | ||||||
|             return of(new mapActions.SetState(state.mapState,urlQueryState)); |           return of(new mapActions.SetState(state.mapState, urlQueryState)); | ||||||
|           }           |  | ||||||
|          }  |  | ||||||
|          return EMPTY;         |  | ||||||
|       })).subscribe((action) => { |  | ||||||
|         if(action) { |  | ||||||
|           this.zone.run(() => { |  | ||||||
|             //console.debug("Url to state"); |  | ||||||
|             this.store.dispatch(action); |  | ||||||
|           }); |  | ||||||
|         } |         } | ||||||
|       }); |       } | ||||||
|  |       return EMPTY; | ||||||
|  |     })).subscribe((action) => { | ||||||
|  |       if (action) { | ||||||
|  |         this.zone.run(() => { | ||||||
|  |           //console.debug("Url to state"); | ||||||
|  |           this.store.dispatch(action); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|     // state to url |     // state to url | ||||||
|  |  | ||||||
|     this.stateSub = this.state$.pipe(switchMap((state) => { |     this.stateSub = this.state$.pipe(switchMap((state) => { | ||||||
|       const newUrl = this.serializeMapState(state.mapState) + "_" + this.serializeService.serialize(state.queryState); |       const newUrl = this.serializeMapState(state.mapState) + "_" + this.serializeService.serialize(state.queryState); | ||||||
|       if(this.lastUrl!=newUrl) { |       if (this.lastUrl != newUrl) { | ||||||
|         this.lastUrl=newUrl; |         this.lastUrl = newUrl; | ||||||
|         return of(state); |         return of(state); | ||||||
|       } |       } | ||||||
|       else { |       else { | ||||||
|         return of(null); |         return of(null); | ||||||
|       } |       } | ||||||
|     })).subscribe((newUrlState: any) => { |     })).subscribe((newUrlState: any) => { | ||||||
|         if(newUrlState) { |       if (newUrlState) { | ||||||
|           //console.debug(`State to url`); |         //console.debug(`State to url`); | ||||||
|           this.replaceUrl(newUrlState.mapState,newUrlState.queryState,newUrlState.replaceUrl); |         this.replaceUrl(newUrlState.mapState, newUrlState.queryState, newUrlState.replaceUrl); | ||||||
|         } |       } | ||||||
|       }); |     }); | ||||||
|     this.initialized = true;    |     this.initialized = true; | ||||||
|     this.showDataLayerSlide$.subscribe((v) => { |     this.showDataLayerSlide$.subscribe((v) => { | ||||||
|       this.dataLayerSlideEnabled=v; |       this.dataLayerSlideEnabled = v; | ||||||
|       this.map.instance.render(); |       this.map.instance.render(); | ||||||
|     }); |     }); | ||||||
|     this.store.select(mapReducers.selectGetViewEnabled).subscribe((v) => { |     this.store.select(mapReducers.selectGetViewEnabled).subscribe((v) => { | ||||||
| @@ -401,7 +406,7 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | |||||||
|     this.store.dispatch(new commonActions.ToggleMenu()); |     this.store.dispatch(new commonActions.ToggleMenu()); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleToggleBaseLayers(event:MouseEvent) { |   handleToggleBaseLayers(event: MouseEvent) { | ||||||
|     this.baseLayersCollapsed = !this.baseLayersCollapsed; |     this.baseLayersCollapsed = !this.baseLayersCollapsed; | ||||||
|     event.preventDefault(); |     event.preventDefault(); | ||||||
|   } |   } | ||||||
| @@ -418,30 +423,30 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   replaceUrl(mapState: IMapState, queryState: IQueryState, replace = true) { |   replaceUrl(mapState: IMapState, queryState: IQueryState, replace = true) { | ||||||
|       if(this.noContent) { |     if (this.noContent) { | ||||||
|         const newMapState = this.serializeMapState(mapState); |       const newMapState = this.serializeMapState(mapState); | ||||||
|         const newQueryState = this.serializeService.serialize(queryState); |       const newQueryState = this.serializeService.serialize(queryState); | ||||||
|         const currentMapState = this.serializeMapState(this.getMapStateFromUrl(this.route.snapshot.paramMap)); |       const currentMapState = this.serializeMapState(this.getMapStateFromUrl(this.route.snapshot.paramMap)); | ||||||
|         const urlQueryState = this.getQueryStateFromUrl(this.route.snapshot.paramMap); |       const urlQueryState = this.getQueryStateFromUrl(this.route.snapshot.paramMap); | ||||||
|         const currentQueryState = urlQueryState==null?"":this.serializeService.serialize(urlQueryState); |       const currentQueryState = urlQueryState == null ? "" : this.serializeService.serialize(urlQueryState); | ||||||
|         if(mapState.baseLayerCode!="" && ((newMapState!= currentMapState)  || (newQueryState!=currentQueryState))) { |       if (mapState.baseLayerCode != "" && ((newMapState != currentMapState) || (newQueryState != currentQueryState))) { | ||||||
|           const parts =["."]; |         const parts = ["."]; | ||||||
|           parts.push(mapState.xCenter.toFixed(5)); |         parts.push(mapState.xCenter.toFixed(5)); | ||||||
|           parts.push(mapState.yCenter.toFixed(5)); |         parts.push(mapState.yCenter.toFixed(5)); | ||||||
|           parts.push( mapState.zoom.toFixed(0)); |         parts.push(mapState.zoom.toFixed(0)); | ||||||
|           parts.push( mapState.rotation.toFixed(2));         |         parts.push(mapState.rotation.toFixed(2)); | ||||||
|           parts.push(mapState.baseLayerCode); |         parts.push(mapState.baseLayerCode); | ||||||
|           parts.push( this.serializeService.serialize(queryState)); |         parts.push(this.serializeService.serialize(queryState)); | ||||||
|            |  | ||||||
|           //console.debug("Replace url",parts); |         //console.debug("Replace url",parts); | ||||||
|           this.router.navigate(parts, { replaceUrl: replace,relativeTo:this.route.parent }); |         this.router.navigate(parts, { replaceUrl: replace, relativeTo: this.route.parent }); | ||||||
|         }   |       } | ||||||
|       }      |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleOnMoveEnd(event) {     |   handleOnMoveEnd(event) { | ||||||
|     if(this.initialized && this.viewEnabled) { |     if (this.initialized && this.viewEnabled) { | ||||||
|       this.zone.run(() =>{ |       this.zone.run(() => { | ||||||
|         //console.debug("Move end"); |         //console.debug("Move end"); | ||||||
|         const map = event.map; |         const map = event.map; | ||||||
|         const view = map.getView(); |         const view = map.getView(); | ||||||
| @@ -461,21 +466,23 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | |||||||
|           } |           } | ||||||
|         }); |         }); | ||||||
|       }); |       }); | ||||||
|     }    |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleOnMouseDown(event: MouseEvent) { |   handleOnMouseDown(event: MouseEvent) { | ||||||
|     event.stopPropagation(); |     event.stopPropagation(); | ||||||
|     this.zone.run(() =>{ |     this.zone.run(() => { | ||||||
|       this.store.dispatch(new commonActions.CloseAll()); |       this.store.dispatch(new commonActions.CloseAll()); | ||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleShowLayerValues(event: MouseEvent) { |   handleShowLayerValues(event: MouseEvent) { | ||||||
|     event.stopPropagation(); |     if (!this.hideShowLayerValues) { | ||||||
|     this.zone.run(() =>{ |       event.stopPropagation(); | ||||||
|       this.store.dispatch(new mapActions.ToggleLayerValuesEnabled()); |       this.zone.run(() => { | ||||||
|     }); |         this.store.dispatch(new mapActions.ToggleLayerValuesEnabled()); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleOnDownload(event) { |   handleOnDownload(event) { | ||||||
| @@ -491,15 +498,15 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleOnToggleVisibility(itemLayer: IItemLayer) { |   handleOnToggleVisibility(itemLayer: IItemLayer) { | ||||||
|     this.store.dispatch(new mapActions.SetVisibility(itemLayer,!itemLayer.visible)); |     this.store.dispatch(new mapActions.SetVisibility(itemLayer, !itemLayer.visible)); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleOnSetOpacity(event:{ layer: IItemLayer,opacity:number }) { |   handleOnSetOpacity(event: { layer: IItemLayer, opacity: number }) { | ||||||
|     this.store.dispatch(new mapActions.SetOpacity(event.layer, event.opacity)); |     this.store.dispatch(new mapActions.SetOpacity(event.layer, event.opacity)); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleZoomToExtent(itemLayer: IItemLayer) { |   handleZoomToExtent(itemLayer: IItemLayer) { | ||||||
|     const extent =  createEmpty(); |     const extent = createEmpty(); | ||||||
|     extend(extent, itemLayer.layer.getExtent()); |     extend(extent, itemLayer.layer.getExtent()); | ||||||
|     if (extent) { |     if (extent) { | ||||||
|       this.store.dispatch(new mapActions.SetExtent(extent)); |       this.store.dispatch(new mapActions.SetExtent(extent)); | ||||||
| @@ -514,21 +521,21 @@ export class MapComponent implements OnInit, OnDestroy,AfterViewInit { | |||||||
|     this.store.dispatch(new mapActions.SelectOverlayLayer(itemLayer)); |     this.store.dispatch(new mapActions.SelectOverlayLayer(itemLayer)); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handlePeriodChange(period:IPeriodState) { |   handlePeriodChange(period: IPeriodState) { | ||||||
|     this.store.dispatch(new mapActions.SetPeriod(period)); |     this.store.dispatch(new mapActions.SetPeriod(period)); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleCitySearch(location:string) { |   handleCitySearch(location: string) { | ||||||
|     this.geolocaterService.geocode(location).subscribe(locations => { |     this.geolocaterService.geocode(location).subscribe(locations => { | ||||||
|       if( locations.length > 0) { |       if (locations.length > 0) { | ||||||
|         const point = new  Point([locations[0].coordinates.lon,locations[0].coordinates.lat]); |         const point = new Point([locations[0].coordinates.lon, locations[0].coordinates.lat]); | ||||||
|         point.transform('EPSG:4326', 'EPSG:3857'); |         point.transform('EPSG:4326', 'EPSG:3857'); | ||||||
|         const circle = new Circle(point.getCoordinates(),5000);//  |         const circle = new Circle(point.getCoordinates(), 5000);//  | ||||||
|         const extent = createEmpty(); |         const extent = createEmpty(); | ||||||
|         extend(extent, circle.getExtent()); |         extend(extent, circle.getExtent()); | ||||||
|         this.store.dispatch(new mapActions.SetExtent(extent)) |         this.store.dispatch(new mapActions.SetExtent(extent)) | ||||||
|       } |       } | ||||||
|      }); |     }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   ngOnDestroy() { |   ngOnDestroy() { | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ export class SelectedItemCropfieldComponent extends AbstractSelectedItemComponen | |||||||
|   public items: Observable<IListItem[]>; |   public items: Observable<IListItem[]>; | ||||||
|  |  | ||||||
|   constructor(store: Store<mapReducers.State | commonReducers.State>, itemTypeService: ItemTypeService, location: Location, router: Router, private itemService$: ItemService,private folderService$: FolderService) { |   constructor(store: Store<mapReducers.State | commonReducers.State>, itemTypeService: ItemTypeService, location: Location, router: Router, private itemService$: ItemService,private folderService$: FolderService) { | ||||||
|     super(store, itemTypeService,location,router); |     super(store, itemTypeService, itemService$, location,router); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   areaInHa(item:IItem):number { |   areaInHa(item:IItem):number { | ||||||
|   | |||||||
| @@ -7,28 +7,31 @@ | |||||||
|         <h2 *ngIf="parentOfItemType('vnd.farmmaps.itemtype.cropfield')">{{parentItem.name}}</h2> |         <h2 *ngIf="parentOfItemType('vnd.farmmaps.itemtype.cropfield')">{{parentItem.name}}</h2> | ||||||
|         <h1>{{item.name}}</h1> |         <h1>{{item.name}}</h1> | ||||||
|       </div> |       </div> | ||||||
|       <div class="legend-container" *ngIf="item?.data.layers;let layers"> |       <ng-container *ngIf="item?.data.layers;let layers"> | ||||||
|         <div class="card menu-card"> |         <div class="legend-container"> | ||||||
|           <div *ngIf="layers.length>1"> |           <div class="card menu-card"> | ||||||
|             <select (change)="onLayerChanged($event.target.value)"> |             <div *ngIf="layers.length>1"> | ||||||
|               <option *ngFor="let l of layers;" [value]="l.index"  [selected]="itemLayer.layerIndex == l.index">{{l.name}}</option> |               <select (change)="onLayerChanged($event.target.value)"> | ||||||
|             </select> |                 <option *ngFor="let l of layers;" [value]="l.index"  [selected]="itemLayer.layerIndex == l.index">{{l.name}}</option> | ||||||
|  |               </select> | ||||||
|  |             </div> | ||||||
|  |             <fm-map-layer-legend [showTitle]="layers.length == 1" | ||||||
|  |               [layer]="layer(layers,itemLayer.layerIndex)" [histogramenabled]="true"></fm-map-layer-legend> | ||||||
|           </div> |           </div> | ||||||
|           <fm-map-layer-legend [showTitle]="layers.length == 1" |  | ||||||
|             [layer]="layer(layers,itemLayer.layerIndex)" [histogramenabled]="true"></fm-map-layer-legend> |  | ||||||
|         </div> |         </div> | ||||||
|       </div> |         <div class="card menu-card"> | ||||||
|       <div class="card menu-card"> |           <ul class="p-0 mt-2"> | ||||||
|         <ul class="p-0 mt-2"> |             <li *ngIf="item.isEditable"><a href="#" class="ms-1 me-1" (click)="handleOnEdit(item)"><i class="fal fa-pencil" aria-hidden="true" i18n-title title="Edit"></i> <span i18n>Edit</span></a></li> | ||||||
|           <li *ngIf="item.isEditable"><a href="#" class="ms-1 me-1" (click)="handleOnEdit(item)"><i class="fal fa-pencil" aria-hidden="true" i18n-title title="Edit"></i> <span i18n>Edit</span></a></li> |             <ng-container *ngIf="itemTypeService.isLayer(item)"> | ||||||
|           <ng-container *ngIf="itemTypeService.isLayer(item)"> |               <li *ngIf="!getItemLayer(item,itemLayer.layerIndex)"><a   href="#" (click)="handleAddAsLayer(item,itemLayer.layerIndex)"><i class="fas fa-layer-plus" aria-hidden="true" i18n-title title="Add as layer"></i> <span i18n>Add as overlay</span></a></li> | ||||||
|             <li *ngIf="!getItemLayer(item,itemLayer.layerIndex)"><a   href="#" (click)="handleAddAsLayer(item,itemLayer.layerIndex)"><i class="fas fa-layer-plus" aria-hidden="true" i18n-title title="Add as layer"></i> <span i18n>Add as overlay</span></a></li> |               <li *ngIf="getItemLayer(item,itemLayer.layerIndex)"><a  href="#" (click)="handleRemoveLayer(item,itemLayer.layerIndex)"><i class="fas fa-layer-minus" aria-hidden="true" i18n-title title="Remove overlay"></i> <span i18n>Remove overlay</span></a></li> | ||||||
|             <li *ngIf="getItemLayer(item,itemLayer.layerIndex)"><a  href="#" (click)="handleRemoveLayer(item,itemLayer.layerIndex)"><i class="fas fa-layer-minus" aria-hidden="true" i18n-title title="Remove overlay"></i> <span i18n>Remove overlay</span></a></li> |             </ng-container> | ||||||
|           </ng-container> |             <li><fm-item-link class="text-primary p-0" [itemCode]="item.code" pathSuffix="data" [showText]="true"></fm-item-link></li> | ||||||
|           <li><fm-item-link class="text-primary p-0" [itemCode]="item.code" pathSuffix="data" [showText]="true"></fm-item-link></li> |             <li><a href="#" (click)="download($event,item,layers,itemLayer.layerIndex)"><i class="fal fa-download" aria-hidden="true" i18n-title title="Download"></i> <span i18n>Download</span></a></li> | ||||||
|         </ul> |           </ul> | ||||||
|       </div> |         </div> | ||||||
|       <fm-map-zoom-to-show-alert [layer]="itemLayer?.layer"></fm-map-zoom-to-show-alert> |         <fm-map-zoom-to-show-alert [layer]="itemLayer?.layer"></fm-map-zoom-to-show-alert> | ||||||
|  |       </ng-container> | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -1,12 +1,13 @@ | |||||||
| import { Component, Injectable } from '@angular/core'; |  | ||||||
| import { Location } from '@angular/common'; | import { Location } from '@angular/common'; | ||||||
| import { Store } from '@ngrx/store'; | import { Component, Injectable, OnDestroy } from '@angular/core'; | ||||||
| import * as mapReducers from '../../reducers/map.reducer'; |  | ||||||
| import { commonReducers, ItemTypeService, ItemService, FolderService } from '@farmmaps/common'; |  | ||||||
| import { Router } from '@angular/router'; | import { Router } from '@angular/router'; | ||||||
|  | import { commonReducers, FolderService, IItem, IItemLinkType, ItemService, ItemTypeService, IUrlType } from '@farmmaps/common'; | ||||||
|  | import { Store } from '@ngrx/store'; | ||||||
|  | import { Subscription } from 'rxjs'; | ||||||
|  | import * as mapActions from '../../actions/map.actions'; | ||||||
|  | import * as mapReducers from '../../reducers/map.reducer'; | ||||||
| import { ForItemType } from '../for-item/for-itemtype.decorator'; | import { ForItemType } from '../for-item/for-itemtype.decorator'; | ||||||
| import { AbstractSelectedItemComponent } from '../selected-item/selected-item.component'; | import { AbstractSelectedItemComponent } from '../selected-item/selected-item.component'; | ||||||
| import * as mapActions from '../../actions/map.actions'; |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ForItemType("vnd.farmmaps.itemtype.geotiff.processed") | @ForItemType("vnd.farmmaps.itemtype.geotiff.processed") | ||||||
| @@ -16,10 +17,15 @@ import * as mapActions from '../../actions/map.actions'; | |||||||
|   templateUrl: './selected-item-geotiff.component.html', |   templateUrl: './selected-item-geotiff.component.html', | ||||||
|   styleUrls: ['./selected-item-geotiff.component.scss'] |   styleUrls: ['./selected-item-geotiff.component.scss'] | ||||||
| }) | }) | ||||||
| export class SelectedItemGeotiffComponent extends AbstractSelectedItemComponent { | export class SelectedItemGeotiffComponent extends AbstractSelectedItemComponent implements OnDestroy { | ||||||
|  |   sub: Subscription; | ||||||
|  |  | ||||||
|   constructor(store: Store<mapReducers.State | commonReducers.State>, itemTypeService: ItemTypeService, location: Location, router: Router, private itemService$: ItemService,private folderService$: FolderService) { |   constructor(store: Store<mapReducers.State | commonReducers.State>, public itemService: ItemService, itemTypeService: ItemTypeService, location: Location, router: Router, private itemService$: ItemService,private folderService$: FolderService) { | ||||||
|     super(store, itemTypeService,location,router); |     super(store, itemTypeService,itemService,location,router); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   ngOnDestroy(): void { | ||||||
|  |     if (this.sub) this.sub.unsubscribe(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   onLayerChanged(layerIndex: number) { |   onLayerChanged(layerIndex: number) { | ||||||
| @@ -29,4 +35,13 @@ export class SelectedItemGeotiffComponent extends AbstractSelectedItemComponent | |||||||
|   layer(layers:any,layerIndex:number) { |   layer(layers:any,layerIndex:number) { | ||||||
|     return layers.find(l => l.index == layerIndex); |     return layers.find(l => l.index == layerIndex); | ||||||
|   } |   } | ||||||
|  |    | ||||||
|  |   download(event:MouseEvent,item:IItem,layers:any,layerIndex:number) { | ||||||
|  |     event.stopPropagation(); | ||||||
|  |     event.preventDefault(); | ||||||
|  |     const itemLink : IItemLinkType = {itemcode:item.code,query:`layer=${this.layer(layers,layerIndex).name}`,pathsuffix:"download", validminutes:10} | ||||||
|  |     this.sub = this.itemService.getItemLink(itemLink).subscribe((itemLinkUrl:IUrlType) => {   | ||||||
|  |         window.location.href = itemLinkUrl.url;      | ||||||
|  |     }) | ||||||
|  |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,25 +7,28 @@ | |||||||
|         <h2 *ngIf="parentOfItemType('vnd.farmmaps.itemtype.cropfield')">{{parentItem.name}}</h2> |         <h2 *ngIf="parentOfItemType('vnd.farmmaps.itemtype.cropfield')">{{parentItem.name}}</h2> | ||||||
|         <h1>{{item.name}}</h1> |         <h1>{{item.name}}</h1> | ||||||
|       </div> |       </div> | ||||||
|       <div class="legend-container" *ngIf="item?.data.layers;let layers"> |       <ng-container *ngIf="item?.data.layers;let layers"> | ||||||
|         <div class="card menu-card"> |         <div class="legend-container"> | ||||||
|           <div *ngIf="layers.length>1"> |           <div class="card menu-card"> | ||||||
|             <select (change)="onLayerChanged($event.target.value)"> |             <div *ngIf="layers.length>1"> | ||||||
|               <option *ngFor="let l of layers;" [value]="l.index" [selected]="itemLayer.layerIndex == l.index">{{l.name}}</option> |               <select (change)="onLayerChanged($event.target.value)"> | ||||||
|             </select> |                 <option *ngFor="let l of layers;" [value]="l.index" [selected]="itemLayer.layerIndex == l.index">{{l.name}}</option> | ||||||
|  |               </select> | ||||||
|  |             </div> | ||||||
|  |             <fm-map-layer-legend [showTitle]="layers.length==1" [layer]="layer(layers,itemLayer.layerIndex)" [histogramenabled]="true"></fm-map-layer-legend> | ||||||
|           </div> |           </div> | ||||||
|           <fm-map-layer-legend [showTitle]="layers.length==1" [layer]="layer(layers,itemLayer.layerIndex)" [histogramenabled]="true"></fm-map-layer-legend> |  | ||||||
|         </div> |         </div> | ||||||
|       </div> |         <div class="card menu-card"> | ||||||
|       <div class="card menu-card"> |           <ul class="p-0 mt-2"> | ||||||
|         <ul class="p-0 mt-2"> |             <li *ngIf="item.isEditable"><a href="#" class="mt-1 me-1" (click)="handleOnEdit(item)" ><i class="fal fa-pencil" aria-hidden="true" i18n-title title="Edit"></i> <span i18n>Edit</span></a></li> | ||||||
|           <li *ngIf="item.isEditable"><a href="#" class="mt-1 me-1" (click)="handleOnEdit(item)" ><i class="fal fa-pencil" aria-hidden="true" i18n-title title="Edit"></i> <span i18n>Edit</span></a></li> |             <ng-container *ngIf="itemTypeService.isLayer(item)"> | ||||||
|           <ng-container *ngIf="itemTypeService.isLayer(item)"> |               <li *ngIf="!getItemLayer(item,itemLayer.layerIndex)"><a   href="#" (click)="handleAddAsLayer(item,itemLayer.layerIndex)" ><i class="fas fa-layer-olus" aria-hidden="true" i18n-title title="Add as layer"></i> <span i18n>Add as overlay</span></a></li> | ||||||
|             <li *ngIf="!getItemLayer(item,itemLayer.layerIndex)"><a   href="#" (click)="handleAddAsLayer(item,itemLayer.layerIndex)" ><i class="fas fa-layer-olus" aria-hidden="true" i18n-title title="Add as layer"></i> <span i18n>Add as overlay</span></a></li> |               <li *ngIf="getItemLayer(item,itemLayer.layerIndex)"><a  href="#" (click)="handleRemoveLayer(item,itemLayer.layerIndex)" ><i class="fas fa-layer-minus" aria-hidden="true" i18n-title title="Remove overlay"></i> <span i18n>Remove overlay</span></a></li> | ||||||
|             <li *ngIf="getItemLayer(item,itemLayer.layerIndex)"><a  href="#" (click)="handleRemoveLayer(item,itemLayer.layerIndex)" ><i class="fas fa-layer-minus" aria-hidden="true" i18n-title title="Remove overlay"></i> <span i18n>Remove overlay</span></a></li> |             </ng-container> | ||||||
|           </ng-container> |             <li><a href="#" (click)="download($event,item,layers,itemLayer.layerIndex)"><i class="fal fa-download" aria-hidden="true" i18n-title title="Download"></i> <span i18n>Download</span></a></li> | ||||||
|         </ul> |           </ul> | ||||||
|       </div> |         </div> | ||||||
|  |       </ng-container> | ||||||
|       <fm-map-zoom-to-show-alert [layer]="itemLayer?.layer"></fm-map-zoom-to-show-alert> |       <fm-map-zoom-to-show-alert [layer]="itemLayer?.layer"></fm-map-zoom-to-show-alert> | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
|   | |||||||
| @@ -1,14 +1,13 @@ | |||||||
| import { Component, Input, Injectable, OnInit } from '@angular/core'; |  | ||||||
| import { Location } from '@angular/common'; | import { Location } from '@angular/common'; | ||||||
| import { Feature } from 'ol'; | import { Component, Injectable, OnDestroy } from '@angular/core'; | ||||||
|  | import { Router } from '@angular/router'; | ||||||
|  | import { commonReducers, FolderService, IItem, IItemLinkType, ItemService, ItemTypeService, IUrlType } from '@farmmaps/common'; | ||||||
| import { Store } from '@ngrx/store'; | import { Store } from '@ngrx/store'; | ||||||
| import * as mapReducers from '../../reducers/map.reducer'; |  | ||||||
| import { commonReducers, ItemTypeService, IItem, Item, ItemService, FolderService, IListItem} from '@farmmaps/common'; |  | ||||||
| import * as mapActions from '../../actions/map.actions'; | import * as mapActions from '../../actions/map.actions'; | ||||||
| import { Router, ActivatedRoute, ParamMap, Event } from '@angular/router'; | import * as mapReducers from '../../reducers/map.reducer'; | ||||||
| import { ForItemType } from '../for-item/for-itemtype.decorator'; | import { ForItemType } from '../for-item/for-itemtype.decorator'; | ||||||
| import { AbstractSelectedItemComponent } from '../selected-item/selected-item.component'; | import { AbstractSelectedItemComponent } from '../selected-item/selected-item.component'; | ||||||
| import { Observable } from 'rxjs'; | import { Subscription } from 'rxjs'; | ||||||
|  |  | ||||||
|  |  | ||||||
| @ForItemType("vnd.farmmaps.itemtype.shape.processed") | @ForItemType("vnd.farmmaps.itemtype.shape.processed") | ||||||
| @@ -18,13 +17,18 @@ import { Observable } from 'rxjs'; | |||||||
|   templateUrl: './selected-item-shape.component.html', |   templateUrl: './selected-item-shape.component.html', | ||||||
|   styleUrls: ['./selected-item-shape.component.scss'] |   styleUrls: ['./selected-item-shape.component.scss'] | ||||||
| }) | }) | ||||||
| export class SelectedItemShapeComponent extends AbstractSelectedItemComponent { | export class SelectedItemShapeComponent extends AbstractSelectedItemComponent implements OnDestroy { | ||||||
|  |  | ||||||
|   constructor(store: Store<mapReducers.State | commonReducers.State>, itemTypeService: ItemTypeService, location: Location, router: Router, private itemService$: ItemService,private folderService$: FolderService) { |  | ||||||
|     super(store, itemTypeService,location,router); |  | ||||||
|   } |  | ||||||
|   public selectedLayer = 0; |   public selectedLayer = 0; | ||||||
|    |   sub: Subscription; | ||||||
|  |  | ||||||
|  |   constructor(store: Store<mapReducers.State | commonReducers.State>, public itemService: ItemService, itemTypeService: ItemTypeService, location: Location, router: Router, private itemService$: ItemService,private folderService$: FolderService) { | ||||||
|  |     super(store, itemTypeService,itemService,location,router); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   ngOnDestroy(): void { | ||||||
|  |     if (this.sub) this.sub.unsubscribe(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   onLayerChanged(layerIndex: number) { |   onLayerChanged(layerIndex: number) { | ||||||
|     this.store.dispatch(new mapActions.SetLayerIndex(layerIndex)); |     this.store.dispatch(new mapActions.SetLayerIndex(layerIndex)); | ||||||
|   } |   } | ||||||
| @@ -32,4 +36,13 @@ export class SelectedItemShapeComponent extends AbstractSelectedItemComponent { | |||||||
|   layer(layers:any,layerIndex:number) { |   layer(layers:any,layerIndex:number) { | ||||||
|     return layers.find(l => l.index == layerIndex); |     return layers.find(l => l.index == layerIndex); | ||||||
|   } |   } | ||||||
|  |      | ||||||
|  |   download(event:MouseEvent,item:IItem,layers:any,layerIndex:number) { | ||||||
|  |     event.stopPropagation(); | ||||||
|  |     event.preventDefault(); | ||||||
|  |     const itemLink : IItemLinkType = {itemcode:item.code,query:`layer=${this.layer(layers,layerIndex).name}`,pathsuffix:"download", validminutes:10} | ||||||
|  |     this.sub = this.itemService.getItemLink(itemLink).subscribe((itemLinkUrl:IUrlType) => {   | ||||||
|  |         window.location.href = itemLinkUrl.url;      | ||||||
|  |     }) | ||||||
|  |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,43 +7,46 @@ | |||||||
|         <h2 *ngIf="parentOfItemType('vnd.farmmaps.itemtype.cropfield')">{{parentItem.name}}</h2> |         <h2 *ngIf="parentOfItemType('vnd.farmmaps.itemtype.cropfield')">{{parentItem.name}}</h2> | ||||||
|         <h1>{{item.name}}</h1> |         <h1>{{item.name}}</h1> | ||||||
|       </div> |       </div> | ||||||
|       <div class="legend-container" *ngIf="item?.data.layers;let layers"> |       <ng-container *ngIf="item?.data.layers;let layers"> | ||||||
|  |         <div class="legend-container"> | ||||||
|  |           <div class="card menu-card"> | ||||||
|  |             <h5><span i18n>Date</span>: {{temporalService.selectedDate(itemLayer)}}</h5> | ||||||
|  |             <div class="d-flex justify-content-between"> | ||||||
|  |               <div> | ||||||
|  |                 <button  *ngIf="temporalService.hasPrevious(itemLayer)" type="button" class="btn btn-link p-0" (click)="handlePreviousTemporal($event)"> | ||||||
|  |                   <i class="fal fa-arrow-circle-left" aria-hidden="true" i18n-title title="Previous"></i> {{ temporalService.previousDate(itemLayer) }} | ||||||
|  |                 </button> | ||||||
|  |               </div> | ||||||
|  |               <div> | ||||||
|  |                 <button  *ngIf="temporalService.hasNext(itemLayer)" type="button" class="btn btn-link p-0" (click)="handleNextTemporal($event)"> | ||||||
|  |                   {{ temporalService.nextDate(itemLayer)}} <i class="fal fa-arrow-circle-right" aria-hidden="true" i18n-title title="Next"></i> | ||||||
|  |                 </button> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |             <fm-map-zoom-to-show-alert [layer]="currentItemLayer()?.layer"></fm-map-zoom-to-show-alert> | ||||||
|  |           </div> | ||||||
|  |           <div class="card menu-card pt-2"> | ||||||
|  |             <div *ngIf="layers.length>1"> | ||||||
|  |               <select (change)="onLayerChanged($event.target.value)"> | ||||||
|  |                 <option *ngFor="let l of layers;" [value]="l.index"  [selected]="itemLayer.layerIndex == l.index">{{l.name}}</option> | ||||||
|  |               </select> | ||||||
|  |             </div> | ||||||
|  |             <fm-map-layer-legend [showTitle]="layers.length == 1" | ||||||
|  |               [layer]="layer(layers,itemLayer.layerIndex)" [histogramenabled]="true"></fm-map-layer-legend> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|         <div class="card menu-card"> |         <div class="card menu-card"> | ||||||
|           <h5><span i18n>Date</span>: {{temporalService.selectedDate(itemLayer)}}</h5> |           <ul class="p-0 mt-2"> | ||||||
|           <div class="d-flex justify-content-between"> |             <li *ngIf="item.isEditable"><a href="#" class="mt-1 me-1" (click)="handleOnEdit(item)"><i class="fal fa-pencil" aria-hidden="true" i18n-title title="Edit"></i> <span i18n>Edit</span></a></li> | ||||||
|             <div> |             <ng-container *ngIf="itemTypeService.isLayer(item)"> | ||||||
|               <button  *ngIf="temporalService.hasPrevious(itemLayer)" type="button" class="btn btn-link p-0" (click)="handlePreviousTemporal($event)"> |               <li *ngIf="!getItemLayer(item,itemLayer.layerIndex)"><a   href="#" (click)="handleAddAsLayer(item,itemLayer.layerIndex)"><i class="fas fa-layer-plus" aria-hidden="true" i18n-title title="Add as layer"></i> <span i18n>Add as overlay</span></a></li> | ||||||
|                 <i class="fal fa-arrow-circle-left" aria-hidden="true" i18n-title title="Previous"></i> {{ temporalService.previousDate(itemLayer) }} |               <li *ngIf="getItemLayer(item,itemLayer.layerIndex)"><a  href="#" (click)="handleRemoveLayer(item,itemLayer.layerIndex)"><i class="fas fa-layer-minus" aria-hidden="true" i18n-title title="Remove overlay"></i> <span i18n>Remove overlay</span></a></li> | ||||||
|               </button> |             </ng-container> | ||||||
|             </div> |             <li><a href="#" (click)="handleGoToChart(item)"><i class="fal fa-line-chart" aria-hidden="true" i18m-title title="Show chart"></i> <span i18n>Show chart</span></a></li> | ||||||
|             <div> |             <li><a href="#" (click)="download($event,item,layers,itemLayer.layerIndex)"><i class="fal fa-download" aria-hidden="true" i18n-title title="Download"></i> <span i18n>Download</span></a></li> | ||||||
|               <button  *ngIf="temporalService.hasNext(itemLayer)" type="button" class="btn btn-link p-0" (click)="handleNextTemporal($event)"> |           </ul> | ||||||
|                 {{ temporalService.nextDate(itemLayer)}} <i class="fal fa-arrow-circle-right" aria-hidden="true" i18n-title title="Next"></i> |  | ||||||
|               </button> |  | ||||||
|             </div> |  | ||||||
|           </div> |  | ||||||
|           <fm-map-zoom-to-show-alert [layer]="currentItemLayer()?.layer"></fm-map-zoom-to-show-alert> |  | ||||||
|         </div> |         </div> | ||||||
|         <div class="card menu-card pt-2"> |       </ng-container> | ||||||
|           <div *ngIf="layers.length>1"> |  | ||||||
|             <select (change)="onLayerChanged($event.target.value)"> |  | ||||||
|               <option *ngFor="let l of layers;" [value]="l.index"  [selected]="itemLayer.layerIndex == l.index">{{l.name}}</option> |  | ||||||
|             </select> |  | ||||||
|           </div> |  | ||||||
|           <fm-map-layer-legend [showTitle]="layers.length == 1" |  | ||||||
|             [layer]="layer(layers,itemLayer.layerIndex)" [histogramenabled]="true"></fm-map-layer-legend> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|       <div class="card menu-card"> |  | ||||||
|         <ul class="p-0 mt-2"> |  | ||||||
|           <li *ngIf="item.isEditable"><a href="#" class="mt-1 me-1" (click)="handleOnEdit(item)"><i class="fal fa-pencil" aria-hidden="true" i18n-title title="Edit"></i> <span i18n>Edit</span></a></li> |  | ||||||
|           <ng-container *ngIf="itemTypeService.isLayer(item)"> |  | ||||||
|             <li *ngIf="!getItemLayer(item,itemLayer.layerIndex)"><a   href="#" (click)="handleAddAsLayer(item,itemLayer.layerIndex)"><i class="fas fa-layer-plus" aria-hidden="true" i18n-title title="Add as layer"></i> <span i18n>Add as overlay</span></a></li> |  | ||||||
|             <li *ngIf="getItemLayer(item,itemLayer.layerIndex)"><a  href="#" (click)="handleRemoveLayer(item,itemLayer.layerIndex)"><i class="fas fa-layer-minus" aria-hidden="true" i18n-title title="Remove overlay"></i> <span i18n>Remove overlay</span></a></li> |  | ||||||
|           </ng-container> |  | ||||||
|           <li><a href="#" (click)="handleGoToChart(item)"><i class="fal fa-line-chart" aria-hidden="true" i18m-title title="Show chart"></i> <span i18n>Show chart</span></a></li> |  | ||||||
|         </ul> |  | ||||||
|       </div> |  | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -1,16 +1,15 @@ | |||||||
| import { Component, Injectable } from '@angular/core'; |  | ||||||
| import { Location } from '@angular/common'; | import { Location } from '@angular/common'; | ||||||
| import { Store } from '@ngrx/store'; | import { Component, Injectable, OnDestroy } from '@angular/core'; | ||||||
| import * as mapReducers from '../../reducers/map.reducer'; |  | ||||||
| import { commonReducers, ItemTypeService, IItem } from '@farmmaps/common'; |  | ||||||
| import { Router } from '@angular/router'; | import { Router } from '@angular/router'; | ||||||
|  | import { commonReducers, IItem, IItemLinkType, ItemService, ItemTypeService, IUrlType } from '@farmmaps/common'; | ||||||
|  | import { Store } from '@ngrx/store'; | ||||||
|  | import { Subscription } from 'rxjs'; | ||||||
|  | import * as mapActions from '../../actions/map.actions'; | ||||||
|  | import { IItemLayer, ITemporalItemLayer } from '../../models/item.layer'; | ||||||
|  | import * as mapReducers from '../../reducers/map.reducer'; | ||||||
|  | import { TemporalService } from '../../services/temporal.service'; | ||||||
| import { ForItemType } from '../for-item/for-itemtype.decorator'; | import { ForItemType } from '../for-item/for-itemtype.decorator'; | ||||||
| import { AbstractSelectedItemComponent } from '../selected-item/selected-item.component'; | import { AbstractSelectedItemComponent } from '../selected-item/selected-item.component'; | ||||||
| import { ITemporalItemLayer} from '../../models/item.layer'; |  | ||||||
| import * as mapActions from '../../actions/map.actions'; |  | ||||||
| import { IItemLayer } from '../../models/item.layer'; |  | ||||||
| import {TemporalService} from '../../services/temporal.service'; |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ForItemType("vnd.farmmaps.itemtype.temporal") | @ForItemType("vnd.farmmaps.itemtype.temporal") | ||||||
| @Injectable() | @Injectable() | ||||||
| @@ -19,10 +18,15 @@ import {TemporalService} from '../../services/temporal.service'; | |||||||
|   templateUrl: './selected-item-temporal.component.html', |   templateUrl: './selected-item-temporal.component.html', | ||||||
|   styleUrls: ['./selected-item-temporal.component.scss'] |   styleUrls: ['./selected-item-temporal.component.scss'] | ||||||
| }) | }) | ||||||
| export class SelectedItemTemporalComponent extends AbstractSelectedItemComponent { | export class SelectedItemTemporalComponent extends AbstractSelectedItemComponent implements OnDestroy { | ||||||
|  |   sub: Subscription; | ||||||
|  |  | ||||||
|   constructor(store: Store<mapReducers.State | commonReducers.State>, itemTypeService: ItemTypeService, location: Location, router: Router,public temporalService:TemporalService) { |   constructor(store: Store<mapReducers.State | commonReducers.State>, public itemService: ItemService, itemTypeService: ItemTypeService, location: Location, router: Router,public temporalService:TemporalService) { | ||||||
|     super(store, itemTypeService,location,router); |     super(store, itemTypeService,itemService,location,router); | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   ngOnDestroy(): void { | ||||||
|  |     if (this.sub) this.sub.unsubscribe(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   onLayerChanged(layerIndex: number) { |   onLayerChanged(layerIndex: number) { | ||||||
| @@ -33,8 +37,6 @@ export class SelectedItemTemporalComponent extends AbstractSelectedItemComponent | |||||||
|     return  (this.itemLayer as ITemporalItemLayer)?.selectedItemLayer |     return  (this.itemLayer as ITemporalItemLayer)?.selectedItemLayer | ||||||
|   } |   } | ||||||
|  |  | ||||||
|    |  | ||||||
|  |  | ||||||
|   handleNextTemporal(event:MouseEvent) { |   handleNextTemporal(event:MouseEvent) { | ||||||
|     this.store.dispatch(new mapActions.NextTemporal()); |     this.store.dispatch(new mapActions.NextTemporal()); | ||||||
|     event.preventDefault();     |     event.preventDefault();     | ||||||
| @@ -61,4 +63,13 @@ export class SelectedItemTemporalComponent extends AbstractSelectedItemComponent | |||||||
|     this.router.navigate(['/viewer', 'temporal', 'item', item.parentCode, new Date(Date.parse(item.dataDate)).getUTCFullYear()]); |     this.router.navigate(['/viewer', 'temporal', 'item', item.parentCode, new Date(Date.parse(item.dataDate)).getUTCFullYear()]); | ||||||
|     return false; |     return false; | ||||||
|   } |   } | ||||||
|  |    | ||||||
|  |   download(event:MouseEvent,item:IItem,layers:any,layerIndex:number) { | ||||||
|  |     event.stopPropagation(); | ||||||
|  |     event.preventDefault(); | ||||||
|  |     const itemLink : IItemLinkType = {itemcode:item.code,query:`layer=${this.layer(layers,layerIndex).name}`,pathsuffix:"download", validminutes:10} | ||||||
|  |     this.sub = this.itemService.getItemLink(itemLink).subscribe((itemLinkUrl:IUrlType) => {   | ||||||
|  |        window.location.href = itemLinkUrl.url;        | ||||||
|  |     }) | ||||||
|  |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ import {Component, Injectable, Input, Directive} from '@angular/core'; | |||||||
| import {Location} from '@angular/common'; | import {Location} from '@angular/common'; | ||||||
| import {Store} from '@ngrx/store'; | import {Store} from '@ngrx/store'; | ||||||
| import * as mapReducers from '../../reducers/map.reducer'; | import * as mapReducers from '../../reducers/map.reducer'; | ||||||
| import {AppConfig, commonReducers, IItem, ItemTypeService} from '@farmmaps/common'; | import {AppConfig, commonReducers, IItem, ItemService, ItemTypeService} from '@farmmaps/common'; | ||||||
| import * as mapActions from '../../actions/map.actions'; | import * as mapActions from '../../actions/map.actions'; | ||||||
| import {Router} from '@angular/router'; | import {Router} from '@angular/router'; | ||||||
| import { IItemLayer } from '../../models/item.layer'; | import { IItemLayer } from '../../models/item.layer'; | ||||||
| @@ -15,7 +15,7 @@ export abstract class AbstractSelectedItemComponent { | |||||||
|   @Input() parentItem: IItem; |   @Input() parentItem: IItem; | ||||||
|   @Input() itemLayer: IItemLayer; |   @Input() itemLayer: IItemLayer; | ||||||
|   @Input() overlayLayers: Array<IItemLayer>; |   @Input() overlayLayers: Array<IItemLayer>; | ||||||
|   constructor(public store: Store<mapReducers.State | commonReducers.State>, public itemTypeService: ItemTypeService, private location: Location, public router: Router) { |   constructor(public store: Store<mapReducers.State | commonReducers.State>, public itemTypeService: ItemTypeService, public itemService: ItemService, private location: Location, public router: Router) { | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleOnView(item: IItem) { |   handleOnView(item: IItem) { | ||||||
| @@ -37,6 +37,11 @@ export abstract class AbstractSelectedItemComponent { | |||||||
|     return false; |     return false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   handleOnDelete(item: IItem) { | ||||||
|  |     this.itemService.deleteItem(item.code); | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   handleAddAsLayer(item: IItem,layerIndex = -1) { |   handleAddAsLayer(item: IItem,layerIndex = -1) { | ||||||
|     this.store.dispatch(new mapActions.AddLayer(item,layerIndex)); |     this.store.dispatch(new mapActions.AddLayer(item,layerIndex)); | ||||||
|     return false; |     return false; | ||||||
| @@ -76,8 +81,8 @@ export abstract class AbstractSelectedItemComponent { | |||||||
| }) | }) | ||||||
| export class SelectedItemComponent extends AbstractSelectedItemComponent { | export class SelectedItemComponent extends AbstractSelectedItemComponent { | ||||||
|  |  | ||||||
|   constructor(store: Store<mapReducers.State | commonReducers.State>, itemTypeService: ItemTypeService, location: Location, router: Router, public config:AppConfig) { |   constructor(store: Store<mapReducers.State | commonReducers.State>, itemTypeService: ItemTypeService, itemService: ItemService, location: Location, router: Router, public config:AppConfig) { | ||||||
|     super(store, itemTypeService,location,router); |     super(store, itemTypeService,itemService,location,router); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   getThumbnailUrl(item:IItem):string { |   getThumbnailUrl(item:IItem):string { | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ import { ILayervalue } from '../models/layer.value'; | |||||||
| import * as mapActions from '../actions/map.actions'; | import * as mapActions from '../actions/map.actions'; | ||||||
| import {commonActions} from '@farmmaps/common'; | import {commonActions} from '@farmmaps/common'; | ||||||
| import { createSelector, createFeatureSelector } from '@ngrx/store'; | import { createSelector, createFeatureSelector } from '@ngrx/store'; | ||||||
|  | import * as _ from 'lodash'; | ||||||
|  |  | ||||||
| import {Feature} from 'ol'; | import {Feature} from 'ol'; | ||||||
| import {Geometry} from 'ol/geom'; | import {Geometry} from 'ol/geom'; | ||||||
| @@ -47,6 +48,7 @@ export interface State { | |||||||
|   query:IQuery, |   query:IQuery, | ||||||
|   parentCode: string, |   parentCode: string, | ||||||
|   features: Array<Feature<Geometry>>, |   features: Array<Feature<Geometry>>, | ||||||
|  |   featuresBackup: Array<Feature<Geometry>>, | ||||||
|   panelVisible: boolean, |   panelVisible: boolean, | ||||||
|   panelCollapsed: boolean, |   panelCollapsed: boolean, | ||||||
|   panelExtraWide: boolean, |   panelExtraWide: boolean, | ||||||
| @@ -93,6 +95,7 @@ export const initialState: State = { | |||||||
|   query: null, |   query: null, | ||||||
|   parentCode: null, |   parentCode: null, | ||||||
|   features: [], |   features: [], | ||||||
|  |   featuresBackup: [], | ||||||
|   panelVisible: false, |   panelVisible: false, | ||||||
|   panelCollapsed: false, |   panelCollapsed: false, | ||||||
|   panelExtraWide: false, |   panelExtraWide: false, | ||||||
| @@ -156,13 +159,15 @@ export function reducer(state = initialState, action: mapActions.Actions | commo | |||||||
|       const a = action as mapActions.StartSearchSuccess; |       const a = action as mapActions.StartSearchSuccess; | ||||||
|       return tassign(state, { |       return tassign(state, { | ||||||
|         features: a.features, |         features: a.features, | ||||||
|  |         featuresBackup: [], | ||||||
|         inSearch:false |         inSearch:false | ||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|     case mapActions.SETFEATURES: { |     case mapActions.SETFEATURES: { | ||||||
|       const a = action as mapActions.SetFeatures; |       const a = action as mapActions.SetFeatures; | ||||||
|       return tassign(state, { |       return tassign(state, { | ||||||
|         features: a.features |         features: a.features, | ||||||
|  |         featuresBackup: [] | ||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|     case mapActions.SELECTFEATURE: { |     case mapActions.SELECTFEATURE: { | ||||||
| @@ -180,6 +185,7 @@ export function reducer(state = initialState, action: mapActions.Actions | commo | |||||||
|         selectedItemLayer: null, |         selectedItemLayer: null, | ||||||
|         showDataLayerSlide: false, |         showDataLayerSlide: false, | ||||||
|         features:[], |         features:[], | ||||||
|  |         featuresBackup: [], | ||||||
|         inSearch:inSearch |         inSearch:inSearch | ||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
| @@ -296,6 +302,7 @@ export function reducer(state = initialState, action: mapActions.Actions | commo | |||||||
|       return tassign(state, { |       return tassign(state, { | ||||||
|         selectedItem: null, |         selectedItem: null, | ||||||
|         features:[], |         features:[], | ||||||
|  |         featuresBackup: [], | ||||||
|         selectedItemLayer:null, |         selectedItemLayer:null, | ||||||
|         searchCollapsed: !panelVisible, |         searchCollapsed: !panelVisible, | ||||||
|         panelVisible: panelVisible, |         panelVisible: panelVisible, | ||||||
| @@ -332,7 +339,8 @@ export function reducer(state = initialState, action: mapActions.Actions | commo | |||||||
|         extent: a.feature.getGeometry().getExtent(), |         extent: a.feature.getGeometry().getExtent(), | ||||||
|         searchCollapsed: false, |         searchCollapsed: false, | ||||||
|         clearEnabled:true, |         clearEnabled:true, | ||||||
|         features:features |         features:features, | ||||||
|  |         featuresBackup:[] | ||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|     case mapActions.UPDATEFEATURESUCCESS: { |     case mapActions.UPDATEFEATURESUCCESS: { | ||||||
| @@ -346,7 +354,7 @@ export function reducer(state = initialState, action: mapActions.Actions | commo | |||||||
|           features.push(state.features[i]); |           features.push(state.features[i]); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       return tassign(state, { features: features }); |       return tassign(state, { features: features, featuresBackup: [] }); | ||||||
|     } |     } | ||||||
|     case mapActions.EXPANDSEARCH: { |     case mapActions.EXPANDSEARCH: { | ||||||
|       return tassign(state, { searchCollapsed: false }); |       return tassign(state, { searchCollapsed: false }); | ||||||
| @@ -391,7 +399,7 @@ export function reducer(state = initialState, action: mapActions.Actions | commo | |||||||
|       return tassign(state, {overlayLayers: [], selectedOverlayLayer: null}); |       return tassign(state, {overlayLayers: [], selectedOverlayLayer: null}); | ||||||
|     } |     } | ||||||
|     case mapActions.CLEARFEATURES: { |     case mapActions.CLEARFEATURES: { | ||||||
|       return tassign(state, {features: [], selectedFeature: null}); |       return tassign(state, {features: [], featuresBackup: [], selectedFeature: null}); | ||||||
|     } |     } | ||||||
|     case mapActions.SETVISIBILITY: { |     case mapActions.SETVISIBILITY: { | ||||||
|       const a = action as mapActions.SetVisibility; |       const a = action as mapActions.SetVisibility; | ||||||
| @@ -509,6 +517,7 @@ export function reducer(state = initialState, action: mapActions.Actions | commo | |||||||
|         searchCollapsed: true, |         searchCollapsed: true, | ||||||
|         searchMinified: false, |         searchMinified: false, | ||||||
|         features: [], |         features: [], | ||||||
|  |         featuresBackup: [], | ||||||
|         query:initialState.query, |         query:initialState.query, | ||||||
|         showLayerSwitcher: false, |         showLayerSwitcher: false, | ||||||
|         extent: null, |         extent: null, | ||||||
| @@ -556,7 +565,8 @@ export function reducer(state = initialState, action: mapActions.Actions | commo | |||||||
|           selectedItem: null, |           selectedItem: null, | ||||||
|           selectedItemLayer: null, |           selectedItemLayer: null, | ||||||
|           showDataLayerSlide: false, |           showDataLayerSlide: false, | ||||||
|           features:[] |           features:[], | ||||||
|  |           featuresBackup:[] | ||||||
|          }); |          }); | ||||||
|       } |       } | ||||||
|       if(state.features.length>0)  { |       if(state.features.length>0)  { | ||||||
| @@ -569,7 +579,7 @@ export function reducer(state = initialState, action: mapActions.Actions | commo | |||||||
|         if(index>=0) { |         if(index>=0) { | ||||||
|           const newFeatures = state.features.slice(0); |           const newFeatures = state.features.slice(0); | ||||||
|           newFeatures.splice(index,1); |           newFeatures.splice(index,1); | ||||||
|           return tassign(state,{features:newFeatures}); |           return tassign(state,{features:newFeatures, featuresBackup:[]}); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       return state; |       return state; | ||||||
| @@ -578,6 +588,17 @@ export function reducer(state = initialState, action: mapActions.Actions | commo | |||||||
|       const a= action as mapActions.SetPanelExtraWide; |       const a= action as mapActions.SetPanelExtraWide; | ||||||
|       return tassign(state,{panelExtraWide:a.panelExtraWide}); |       return tassign(state,{panelExtraWide:a.panelExtraWide}); | ||||||
|     } |     } | ||||||
|  |     case mapActions.BACKUPFEATURES: { | ||||||
|  |       return tassign(state, { | ||||||
|  |         featuresBackup: _.cloneDeep(state.features) | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |     case mapActions.RESTOREFEATURES: { | ||||||
|  |       return tassign(state, { | ||||||
|  |         features: _.cloneDeep(state.featuresBackup), | ||||||
|  |         featuresBackup: [] | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|     default: { |     default: { | ||||||
|       return state; |       return state; | ||||||
|     } |     } | ||||||
| @@ -637,6 +658,8 @@ export const selectGetExtent = createSelector(selectMapState, getExtent); | |||||||
| export const selectGetViewExtent = createSelector(selectMapState, getViewExtent); | export const selectGetViewExtent = createSelector(selectMapState, getViewExtent); | ||||||
| export const selectGetOverlayLayers = createSelector(selectMapState, getOverlayLayers); | export const selectGetOverlayLayers = createSelector(selectMapState, getOverlayLayers); | ||||||
| export const selectGetBaseLayers = createSelector(selectMapState, getBaseLayers); | export const selectGetBaseLayers = createSelector(selectMapState, getBaseLayers); | ||||||
|  | export const selectGetBaseMaps = createSelector(selectGetBaseLayers, (layers) => layers.filter(layer => layer.item.data.category === "baseMap")); | ||||||
|  | export const selectGetArealMaps = createSelector(selectGetBaseLayers, (layers) => layers.filter(layer => layer.item.data.category === "aerialPhoto").sort((a, b) => a.item.dataDate <= b.item.dataDate ? 1 : -1)); | ||||||
| export const selectGetProjection = createSelector(selectMapState, getProjection); | export const selectGetProjection = createSelector(selectMapState, getProjection); | ||||||
| export const selectGetSelectedBaseLayer = createSelector(selectMapState, getSelectedBaseLayer); | export const selectGetSelectedBaseLayer = createSelector(selectMapState, getSelectedBaseLayer); | ||||||
| export const selectGetSelectedOverlayLayer = createSelector(selectMapState, getSelectedOverlayLayer); | export const selectGetSelectedOverlayLayer = createSelector(selectMapState, getSelectedOverlayLayer); | ||||||
|   | |||||||
| @@ -8,8 +8,8 @@ | |||||||
|     "tslib": "^2.0.0" |     "tslib": "^2.0.0" | ||||||
|   }, |   }, | ||||||
|   "peerDependencies": { |   "peerDependencies": { | ||||||
|     "@angular/common": "^18.2.3", |     "@angular/common": "18.2.3", | ||||||
|     "@angular/core":  "^18.2.3", |     "@angular/core":  "18.2.3", | ||||||
|     "cesium": "^1.97.0", |     "cesium": "^1.97.0", | ||||||
|     "ol-cesium": ">=2.13.0" |     "ol-cesium": ">=2.13.0" | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -6,9 +6,9 @@ | |||||||
|   }, |   }, | ||||||
|   "peerDependencies": { |   "peerDependencies": { | ||||||
|     "@ng-bootstrap/ng-bootstrap": "^17.0.1", |     "@ng-bootstrap/ng-bootstrap": "^17.0.1", | ||||||
|     "@angular/common": "^18.2.3", |     "@angular/common": "18.2.3", | ||||||
|     "@angular/core": "^18.2.3", |     "@angular/core": "18.2.3", | ||||||
|     "@angular/forms": "^18.2.3", |     "@angular/forms": "18.2.3", | ||||||
|     "@ngrx/effects": "^18.0.2", |     "@ngrx/effects": "^18.0.2", | ||||||
|     "@ngrx/router-store": "^18.0.2", |     "@ngrx/router-store": "^18.0.2", | ||||||
|     "@ngrx/store": "^18.0.2", |     "@ngrx/store": "^18.0.2", | ||||||
|   | |||||||
| @@ -83,6 +83,11 @@ export const SETPAGEMODE = '[AppCommon] SetPageMode'; | |||||||
|  |  | ||||||
| export const SETUNREADNOTIFICATIONS = '[AppCommon] SetUnreadNotifications'; | export const SETUNREADNOTIFICATIONS = '[AppCommon] SetUnreadNotifications'; | ||||||
|  |  | ||||||
|  | export const SWITCHLANGUAGE = '[AppCommon] SwitchLanguage'; | ||||||
|  |  | ||||||
|  | export const SETSETTINGMENUBACKGROUNDCOLOR = '[AppCommon] SetSettingMenuBackgroundColor'; | ||||||
|  |  | ||||||
|  |  | ||||||
| export class InitUser implements Action { | export class InitUser implements Action { | ||||||
|   readonly type = INITUSER; |   readonly type = INITUSER; | ||||||
|  |  | ||||||
| @@ -371,6 +376,17 @@ export class SetUnreadNotifications implements Action { | |||||||
|  |  | ||||||
| } | } | ||||||
|    |    | ||||||
|  | export class SwitchLanguage implements Action { | ||||||
|  |   readonly type = SWITCHLANGUAGE; | ||||||
|  |  | ||||||
|  |   constructor(public locale:string) { } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export class SetSettingMenuBackgroundColor implements Action { | ||||||
|  |   readonly type = SETSETTINGMENUBACKGROUNDCOLOR; | ||||||
|  |  | ||||||
|  |   constructor(public color:string) { } | ||||||
|  | } | ||||||
|  |  | ||||||
| export type Actions = OpenModal | export type Actions = OpenModal | ||||||
|   | InitRoot |   | InitRoot | ||||||
| @@ -418,6 +434,8 @@ export type Actions = OpenModal | |||||||
|   | ToggleHelpMenu |   | ToggleHelpMenu | ||||||
|   | ToggleSettingMenu |   | ToggleSettingMenu | ||||||
|   | NotificationEvent |   | NotificationEvent | ||||||
|   | SetUnreadNotifications; |   | SetUnreadNotifications | ||||||
|  |   | SwitchLanguage | ||||||
|  |   | SetSettingMenuBackgroundColor; | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -13,6 +13,7 @@ import { SchemaService } from './services/schema.service'; | |||||||
| import { FolderService } from './services/folder.service'; | import { FolderService } from './services/folder.service'; | ||||||
| import { TimespanService } from './services/timespan.service'; | import { TimespanService } from './services/timespan.service'; | ||||||
| import { ItemService } from './services/item.service'; | import { ItemService } from './services/item.service'; | ||||||
|  | import { TaskService } from './services/task.service'; | ||||||
| import { AdminService } from './services/admin.service'; | import { AdminService } from './services/admin.service'; | ||||||
| import { EventService } from './services/event.service'; | import { EventService } from './services/event.service'; | ||||||
| import { TypeaheadService } from './services/typeahead.service'; | import { TypeaheadService } from './services/typeahead.service'; | ||||||
| @@ -44,6 +45,7 @@ export { | |||||||
|   ItemTypeService, |   ItemTypeService, | ||||||
|   TimespanService, |   TimespanService, | ||||||
|   ItemService, |   ItemService, | ||||||
|  |   TaskService, | ||||||
|   AdminService, |   AdminService, | ||||||
|   EventService, |   EventService, | ||||||
|   TypeaheadService, |   TypeaheadService, | ||||||
|   | |||||||
| @@ -33,7 +33,7 @@ | |||||||
|     <fm-resumable-file-upload></fm-resumable-file-upload> |     <fm-resumable-file-upload></fm-resumable-file-upload> | ||||||
|   </ng-container>   |   </ng-container>   | ||||||
|   <div class="user-menu apponly"> |   <div class="user-menu apponly"> | ||||||
|     <fm-setting-menu [user]="user|async" [showMenu]="settingMenuVisible|async"></fm-setting-menu> |     <fm-setting-menu [user]="user|async" [showMenu]="settingMenuVisible|async" [backgroundColor]="settingMenuBackgroundColor|async"></fm-setting-menu> | ||||||
|     <fm-help-menu [user]="user|async" [showMenu]="helpMenuVisible|async"></fm-help-menu> |     <fm-help-menu [user]="user|async" [showMenu]="helpMenuVisible|async"></fm-help-menu> | ||||||
|     <fm-notification-menu [user]="user|async" [unread]="unreadNotifications|async" [showMenu]="notificationMenuVisible|async"></fm-notification-menu> |     <fm-notification-menu [user]="user|async" [unread]="unreadNotifications|async" [showMenu]="notificationMenuVisible|async"></fm-notification-menu> | ||||||
|     <fm-app-menu [user]="user|async" [showMenu]="appMenuVisible|async"></fm-app-menu> |     <fm-app-menu [user]="user|async" [showMenu]="appMenuVisible|async"></fm-app-menu> | ||||||
|   | |||||||
| @@ -50,6 +50,7 @@ export class AppComponent implements OnInit, OnDestroy { | |||||||
|   public unreadNotifications: Observable<number> = this.store$.select(appReducers.SelectgetUnreadNotifications); |   public unreadNotifications: Observable<number> = this.store$.select(appReducers.SelectgetUnreadNotifications); | ||||||
|   public user: Observable<IUser> = this.store$.select(appReducers.SelectGetUser); |   public user: Observable<IUser> = this.store$.select(appReducers.SelectGetUser); | ||||||
|   public isPageMode: Observable<boolean> = this.store$.select(appReducers.SelectGetIsPageMode); |   public isPageMode: Observable<boolean> = this.store$.select(appReducers.SelectGetIsPageMode); | ||||||
|  |   public settingMenuBackgroundColor: Observable<string> = this.store$.select(appReducers.SelectGetSettingMenuBackgroundColor); | ||||||
|   @Input() showUploadProgress = true; |   @Input() showUploadProgress = true; | ||||||
|  |  | ||||||
|   constructor( |   constructor( | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ | |||||||
|         <i class="fal fa-image"></i> |         <i class="fal fa-image"></i> | ||||||
|         <div i18n>No image selected</div> |         <div i18n>No image selected</div> | ||||||
|       </div> |       </div> | ||||||
|       <image-cropper #imageCropper |       <image-cropper #imageCropper output="base64"  | ||||||
|       [imageChangedEvent]="imageChangedEvent" |       [imageChangedEvent]="imageChangedEvent" | ||||||
|       [maintainAspectRatio]="true" |       [maintainAspectRatio]="true" | ||||||
|       [format]="imageType" |       [format]="imageType" | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ div.resumable-file-upload { | |||||||
|   bottom: 0px; |   bottom: 0px; | ||||||
|   width: 300px; |   width: 300px; | ||||||
|   max-height: 250px; |   max-height: 250px; | ||||||
|  |   z-index: 1; | ||||||
|   /*z-index:2000 !important;*/ |   /*z-index:2000 !important;*/ | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| <div> | <div> | ||||||
|     <div (click)="toggle($event)" class="rounded-circle menu-button hidden"  [ngClass]="{'hidden':!user || noContent}"> |     <div (click)="toggle($event)" class="rounded-circle menu-button hidden" [style.background-color]="backgroundColor" [ngClass]="{'hidden':!user || noContent}"> | ||||||
|         <span i18n-title title="Settings"><i class="fas fa-gear" aria-hidden="true"></i></span> |         <span i18n-title title="Settings"><i class="fas fa-gear" aria-hidden="true"></i></span> | ||||||
|         <div class="menu hidden" [ngClass]="{'hidden':!showMenu}">                                      |         <div class="menu hidden" [ngClass]="{'hidden':!showMenu}">                                      | ||||||
|             <router-outlet name="setting-menu" (activate)="activateRoute()" (deactivate)="deActivateRoute()"></router-outlet>             |             <router-outlet name="setting-menu" (activate)="activateRoute()" (deactivate)="deActivateRoute()"></router-outlet>             | ||||||
|   | |||||||
| @@ -15,6 +15,7 @@ export class SettingMenuComponent implements OnInit { | |||||||
|    |    | ||||||
|   @Input() user:IUser; |   @Input() user:IUser; | ||||||
|   @Input() showMenu:boolean; |   @Input() showMenu:boolean; | ||||||
|  |   @Input() backgroundColor:string; | ||||||
|   public noContent = true; |   public noContent = true; | ||||||
|  |  | ||||||
|   constructor(private store: Store<appReducers.State>) { } |   constructor(private store: Store<appReducers.State>) { } | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { Injectable } from '@angular/core'; | import { Injectable, Inject, LOCALE_ID } from '@angular/core'; | ||||||
| import { Router } from '@angular/router'; | import { Router } from '@angular/router'; | ||||||
| import { OAuthService,UserInfo } from 'angular-oauth2-oidc'; | import { OAuthService,UserInfo } from 'angular-oauth2-oidc'; | ||||||
| import { Store } from '@ngrx/store'; | import { Store } from '@ngrx/store'; | ||||||
| @@ -16,6 +16,7 @@ import {StateSerializerService} from '../services/state-serializer.service'; | |||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class AppCommonEffects { | export class AppCommonEffects { | ||||||
|  |   locale: string; | ||||||
|  |  | ||||||
|   login$ =  createEffect(() => this.actions$.pipe( |   login$ =  createEffect(() => this.actions$.pipe( | ||||||
|     ofType(appCommonActions.LOGIN), |     ofType(appCommonActions.LOGIN), | ||||||
| @@ -47,11 +48,27 @@ export class AppCommonEffects { | |||||||
|     first(), |     first(), | ||||||
|     switchMap((action) => { |     switchMap((action) => { | ||||||
|         return zip(this.userService$.getCurrentUser(),from(this.oauthService$.loadUserProfile())).pipe( |         return zip(this.userService$.getCurrentUser(),from(this.oauthService$.loadUserProfile())).pipe( | ||||||
|         switchMap(([user,userInfo]) => {return of(new appCommonActions.InitUserSuccess(user,userInfo as UserInfo))} ), |         switchMap(([user,userInfo]) => { | ||||||
|  |           if (location.hostname === 'localhost' || user.language === undefined || user.language === this.locale)  | ||||||
|  |           { | ||||||
|  |             return of(new appCommonActions.InitUserSuccess(user,userInfo as UserInfo)) | ||||||
|  |           } | ||||||
|  |           return of(new appCommonActions.SwitchLanguage(user.language)) | ||||||
|  |         }), | ||||||
|         catchError(error => of(new appCommonActions.Fail(error)))) |         catchError(error => of(new appCommonActions.Fail(error)))) | ||||||
|     } |     } | ||||||
|   ))); |   ))); | ||||||
|      |  | ||||||
|  |   switchLanguage$ = createEffect(() => this.actions$.pipe( | ||||||
|  |     ofType(appCommonActions.SWITCHLANGUAGE), | ||||||
|  |     switchMap((action) => { | ||||||
|  |       const a = action as appCommonActions.SwitchLanguage; | ||||||
|  |       if (location.href.includes(`/${this.locale}/`)) { | ||||||
|  |         const newPath = location.href.replace(`/${this.locale}/`, `/${a.locale}/`); | ||||||
|  |         location.replace(newPath); | ||||||
|  |       } | ||||||
|  |       return of(undefined); | ||||||
|  |     })),{ dispatch:false}); | ||||||
|  |  | ||||||
|   initUserPackages$ =  createEffect(() => this.actions$.pipe( |   initUserPackages$ =  createEffect(() => this.actions$.pipe( | ||||||
|     ofType(appCommonActions.INITUSERPACKAGES), |     ofType(appCommonActions.INITUSERPACKAGES), | ||||||
| @@ -134,15 +151,22 @@ export class AppCommonEffects { | |||||||
|       withLatestFrom(this.store$.select(appCommonReducers.selectGetItemTypes)), |       withLatestFrom(this.store$.select(appCommonReducers.selectGetItemTypes)), | ||||||
|       switchMap(([action, itemtypes]) => { |       switchMap(([action, itemtypes]) => { | ||||||
|         const a = action as appCommonActions.EditItem; |         const a = action as appCommonActions.EditItem; | ||||||
|         var editor =  "property"; |         var editor = "property"; | ||||||
|         if(a.item.itemType) { |         if(a.item.itemType) { | ||||||
|           const itemType = itemtypes[a.item.itemType]; |           const itemType = itemtypes[a.item.itemType]; | ||||||
|           var editor = itemType && itemType.editor ? itemType.editor : editor;   |           var editor = itemType && itemType.editor ? itemType.editor : editor;   | ||||||
|         } |         } | ||||||
|       this.router$.navigate(['/editor',editor,'item', a.item.code]) |         if (editor == 'croppingscheme') {  | ||||||
|       return []; |           // Exeception: no 'item' in route | ||||||
|     } |           this.router$.navigate(['/editor', editor, a.item.code]) | ||||||
|   ))); |         } | ||||||
|  |         else { | ||||||
|  |           this.router$.navigate(['/editor', editor, 'item', a.item.code]) | ||||||
|  |         } | ||||||
|  |         return []; | ||||||
|  |       } | ||||||
|  |     ) | ||||||
|  |   )); | ||||||
|  |  | ||||||
|   viewItem$ =  createEffect(() => this.actions$.pipe( |   viewItem$ =  createEffect(() => this.actions$.pipe( | ||||||
|     ofType(appCommonActions.VIEWITEM), |     ofType(appCommonActions.VIEWITEM), | ||||||
| @@ -201,7 +225,8 @@ export class AppCommonEffects { | |||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   constructor(private actions$: Actions, private store$: Store<appCommonReducers.State>, private oauthService$: OAuthService, private itemService$: ItemService, private folderService$:FolderService, private userService$: UserService, private router$: Router, private stateSerializerService$:StateSerializerService) { |   constructor(private actions$: Actions, private store$: Store<appCommonReducers.State>, private oauthService$: OAuthService, private itemService$: ItemService, private folderService$:FolderService, private userService$: UserService, private router$: Router, private stateSerializerService$:StateSerializerService, @Inject(LOCALE_ID) locale: string) { | ||||||
|  |     this.locale = locale; | ||||||
|     store$.dispatch(new appCommonActions.LoadItemTypes()); |     store$.dispatch(new appCommonActions.LoadItemTypes()); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @@ -26,6 +26,7 @@ export class Item implements IItem { | |||||||
|   public data?:any; |   public data?:any; | ||||||
|   public isEditable?:boolean; |   public isEditable?:boolean; | ||||||
|   public owner?: string; |   public owner?: string; | ||||||
|  |   public fullPath?: string; | ||||||
|  |  | ||||||
|   constructor() {   |   constructor() {   | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
|  |  | ||||||
| export interface IItemTask  { | export interface IItemTask  { | ||||||
|   code?: string; |   code?: string; | ||||||
|  |   workflowCode?: string; | ||||||
|   taskType?: string; |   taskType?: string; | ||||||
|   attributes?: any; |   attributes?: any; | ||||||
|   message?: string, |   message?: string, | ||||||
| @@ -11,6 +12,7 @@ export interface IItemTask  { | |||||||
|  |  | ||||||
| export class ItemTask implements IItemTask { | export class ItemTask implements IItemTask { | ||||||
|   public code?:string; |   public code?:string; | ||||||
|  |   public workflowCode?: string; | ||||||
|   public taskType?: string; |   public taskType?: string; | ||||||
|   public attributes?: any; |   public attributes?: any; | ||||||
|   public message?: string; |   public message?: string; | ||||||
| @@ -18,6 +20,6 @@ export class ItemTask implements IItemTask { | |||||||
|   public started?: Date; |   public started?: Date; | ||||||
|   public finished?: Date; |   public finished?: Date; | ||||||
|  |  | ||||||
|   constructor() {   |   constructor() { | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -29,7 +29,8 @@ export interface State { | |||||||
|   settingMenuVisible: boolean, |   settingMenuVisible: boolean, | ||||||
|   unreadNotifications: number, |   unreadNotifications: number, | ||||||
|   isOnline: boolean, |   isOnline: boolean, | ||||||
|   isPageMode:boolean |   isPageMode:boolean, | ||||||
|  |   settingMenuBackgroundColor:string | ||||||
| } | } | ||||||
|  |  | ||||||
| export const initialState: State = { | export const initialState: State = { | ||||||
| @@ -51,7 +52,8 @@ export const initialState: State = { | |||||||
|   settingMenuVisible: false, |   settingMenuVisible: false, | ||||||
|   unreadNotifications: 0, |   unreadNotifications: 0, | ||||||
|   isOnline: true, |   isOnline: true, | ||||||
|   isPageMode: true |   isPageMode: true, | ||||||
|  |   settingMenuBackgroundColor:'rgb(128, 128, 128)' | ||||||
| } | } | ||||||
|  |  | ||||||
| export function reducer(state = initialState, action: appCommonActions.Actions ): State { | export function reducer(state = initialState, action: appCommonActions.Actions ): State { | ||||||
| @@ -187,6 +189,10 @@ export function reducer(state = initialState, action: appCommonActions.Actions ) | |||||||
|       const a = action as appCommonActions.SetUnreadNotifications; |       const a = action as appCommonActions.SetUnreadNotifications; | ||||||
|       return tassign(state,{unreadNotifications:a.unread}); |       return tassign(state,{unreadNotifications:a.unread}); | ||||||
|     } |     } | ||||||
|  |     case appCommonActions.SETSETTINGMENUBACKGROUNDCOLOR: { | ||||||
|  |       const a = action as appCommonActions.SetSettingMenuBackgroundColor; | ||||||
|  |       return tassign(state,{settingMenuBackgroundColor:a.color}); | ||||||
|  |     }     | ||||||
|     default: { |     default: { | ||||||
|       return state; |       return state; | ||||||
|     } |     } | ||||||
| @@ -212,6 +218,7 @@ export const getSettingMenuVisible = (state: State) => state.settingMenuVisible; | |||||||
| export const getUnreadNotifications = (state: State) => state.unreadNotifications; | export const getUnreadNotifications = (state: State) => state.unreadNotifications; | ||||||
| export const getIsOnline = (state: State) => state.isOnline; | export const getIsOnline = (state: State) => state.isOnline; | ||||||
| export const getIsPageMode = (state: State) => state.isPageMode; | export const getIsPageMode = (state: State) => state.isPageMode; | ||||||
|  | export const getSettingMenuBackgroundColor = (state: State) => state.settingMenuBackgroundColor; | ||||||
|  |  | ||||||
| export const selectAppCommonState = createFeatureSelector<State>(MODULE_NAME); | export const selectAppCommonState = createFeatureSelector<State>(MODULE_NAME); | ||||||
|  |  | ||||||
| @@ -238,5 +245,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); | ||||||
|  | export const SelectGetSettingMenuBackgroundColor= createSelector(selectAppCommonState,getSettingMenuBackgroundColor); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -52,4 +52,20 @@ export class AdminService { | |||||||
|   postItemTask(item: IItem, task: IItemTask): Observable<IItemTask> { |   postItemTask(item: IItem, task: IItemTask): Observable<IItemTask> { | ||||||
|     return this.httpClient.post<IItemTask>(`${this.ApiEndpoint()}/api/v1/admin/${item.code}/tasks`, task); |     return this.httpClient.post<IItemTask>(`${this.ApiEndpoint()}/api/v1/admin/${item.code}/tasks`, task); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   getFileStorageOverview(): Observable<any> { | ||||||
|  |     return this.httpClient.get<any>(`${this.ApiEndpoint()}/api/v1/admin/filestorage/folderinfo/overview`); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   getFileStorageInfo(user: string): Observable<any> { | ||||||
|  |     return this.httpClient.get<any>(`${this.ApiEndpoint()}/api/v1/admin/filestorage/folderinfo/${user}`); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   getFileStorageLog(): Observable<any> { | ||||||
|  |     return this.httpClient.get<any>(`${this.ApiEndpoint()}/api/v1/admin/filestorage/folderinfo/log`); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   validateEmail(user: IUser) { | ||||||
|  |     return this.httpClient.post(`${this.ApiEndpoint()}/api/v1/admin/user/${user.code}/validateemail`, user); | ||||||
|  |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,14 +1,12 @@ | |||||||
| import { Injectable } from '@angular/core'; | import {Injectable} from '@angular/core'; | ||||||
| import { Observable } from 'rxjs'; | import {Observable} from 'rxjs'; | ||||||
| import { map } from 'rxjs/operators'; | import {IItemType} from '../models/item.type'; | ||||||
| import { IItemType } from '../models/item.type'; | import {IItemLinkType} from '../models/itemlink.type'; | ||||||
| import { IItemLinkType } from '../models/itemlink.type'; | import {IItem} from '../models/item'; | ||||||
| import { IUrlType } from '../models/url.type'; | import {IJsonline} from '../models/json-line'; | ||||||
| import { IItem } from '../models/item'; | import {IItemTask} from '../models/itemTask'; | ||||||
| import { IJsonline } from '../models/json-line'; | import {HttpClient, HttpHeaders, HttpParams} from '@angular/common/http'; | ||||||
| import { IItemTask } from '../models/itemTask'; | import {AppConfig} from '../shared/app.config'; | ||||||
| import { HttpClient, HttpParams, HttpHeaders } from "@angular/common/http"; |  | ||||||
| import { AppConfig } from "../shared/app.config"; |  | ||||||
| import {ItemTypeService} from './itemtype.service'; | import {ItemTypeService} from './itemtype.service'; | ||||||
| import {IListItem} from '../models/list.item'; | import {IListItem} from '../models/list.item'; | ||||||
|  |  | ||||||
| @@ -81,7 +79,8 @@ export class ItemService { | |||||||
|  |  | ||||||
|   getItemList(itemType?: string, dataFilter?: any, level?: number, atItemLocationItemCode?: string, |   getItemList(itemType?: string, dataFilter?: any, level?: number, atItemLocationItemCode?: string, | ||||||
|     indexed?: boolean, validToday?: boolean,tags?:string,crs?:string, startDate?: Date, endDate?: Date, |     indexed?: boolean, validToday?: boolean,tags?:string,crs?:string, startDate?: Date, endDate?: Date, | ||||||
|     skip?: number, take?: number, sourceTask?: string, exactMatchStartOrEndDate?: boolean): Observable<IItem[]> { |     skip?: number, take?: number, sourceTask?: string, exactMatchStartOrEndDate?: boolean, owner?:string, | ||||||
|  |     includeFullpath?: boolean): Observable<IItem[]> { | ||||||
|     let params = new HttpParams(); |     let params = new HttpParams(); | ||||||
|     if(itemType)  params = params.append("it", itemType); |     if(itemType)  params = params.append("it", itemType); | ||||||
|     if(dataFilter) params = params.append("df", JSON.stringify(dataFilter)); |     if(dataFilter) params = params.append("df", JSON.stringify(dataFilter)); | ||||||
| @@ -93,16 +92,17 @@ export class ItemService { | |||||||
|     if (crs) params = params.append("crs", crs); |     if (crs) params = params.append("crs", crs); | ||||||
|     if (startDate) params = params.append("sDate", startDate.toISOString()); |     if (startDate) params = params.append("sDate", startDate.toISOString()); | ||||||
|     if (endDate) params = params.append("eDate", endDate.toISOString()); |     if (endDate) params = params.append("eDate", endDate.toISOString()); | ||||||
|     if (endDate) params = params.append("eDate", endDate.toISOString()); |  | ||||||
|     if (sourceTask) params = params.append("sourceTask", sourceTask); |     if (sourceTask) params = params.append("sourceTask", sourceTask); | ||||||
|     if(skip) params = params.append("skip", skip); |     if(skip) params = params.append("skip", skip); | ||||||
|     if(take) params = params.append("take", take); |     if(take) params = params.append("take", take); | ||||||
|     if(exactMatchStartOrEndDate !== undefined) params = params.append("exactMatchStartOrEndDate", exactMatchStartOrEndDate);     |     if(exactMatchStartOrEndDate !== undefined) params = params.append("exactMatchStartOrEndDate", exactMatchStartOrEndDate); | ||||||
|  |     if(owner) params = params.append("owner", owner); | ||||||
|  |     if(includeFullpath) params = params.append("includeFullpath",includeFullpath?"true":"false"); | ||||||
|     return this.httpClient.get<IItem[]>(`${this.ApiEndpoint()}/api/v1/items/`, { params: params }); |     return this.httpClient.get<IItem[]>(`${this.ApiEndpoint()}/api/v1/items/`, { params: params }); | ||||||
|   } |   } | ||||||
|    |  | ||||||
|   getItemListCount(itemType: string, startDate?: Date, endDate?: Date, sourceTask?: string, |   getItemListCount(itemType: string, startDate?: Date, endDate?: Date, sourceTask?: string, | ||||||
|     indexed?: boolean, exactMatchStartOrEndDate?: boolean): Observable<number> { |     indexed?: boolean, exactMatchStartOrEndDate?: boolean, owner?:string): Observable<number> { | ||||||
|     let params = new HttpParams(); |     let params = new HttpParams(); | ||||||
|     params = params.append("it", itemType); |     params = params.append("it", itemType); | ||||||
|     if (sourceTask) params = params.append("sourceTask", sourceTask); |     if (sourceTask) params = params.append("sourceTask", sourceTask); | ||||||
| @@ -110,12 +110,14 @@ export class ItemService { | |||||||
|     if (endDate) params = params.append("eDate", endDate.toISOString()); |     if (endDate) params = params.append("eDate", endDate.toISOString()); | ||||||
|     if(indexed) params = params.append("ind",indexed?"true":"false"); |     if(indexed) params = params.append("ind",indexed?"true":"false"); | ||||||
|     if(exactMatchStartOrEndDate !== undefined) params = params.append("exactMatchStartOrEndDate", exactMatchStartOrEndDate); |     if(exactMatchStartOrEndDate !== undefined) params = params.append("exactMatchStartOrEndDate", exactMatchStartOrEndDate); | ||||||
|  |     if(owner) params = params.append("owner", owner); | ||||||
|     return this.httpClient.get<number>(`${this.ApiEndpoint()}/api/v1/items/count`, { params: params }); |     return this.httpClient.get<number>(`${this.ApiEndpoint()}/api/v1/items/count`, { params: params }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   getChildItemList(parentcode: string, itemType?: string, dataFilter?: any, level = 1, deep = true, |   getChildItemList(parentcode: string, itemType?: string, dataFilter?: any, level = 1, deep = true, | ||||||
|                    startDate?: Date, endDate?: Date, skip?: number, take?: number, |                    startDate?: Date, endDate?: Date, skip?: number, take?: number, | ||||||
|                    exactMatchStartOrEndDate?: boolean): Observable<IItem[]> { |                    exactMatchStartOrEndDate?: boolean, owner?:string, indexed?: boolean, | ||||||
|  |                   atItemLocationItemCode?: string): Observable<IItem[]> { | ||||||
|     let params = new HttpParams(); |     let params = new HttpParams(); | ||||||
|     if(itemType != null) { |     if(itemType != null) { | ||||||
|       params = params.append("it", itemType); |       params = params.append("it", itemType); | ||||||
| @@ -127,15 +129,19 @@ export class ItemService { | |||||||
|     params = params.append("deep", deep.toString()); |     params = params.append("deep", deep.toString()); | ||||||
|     if (startDate) params = params.append("sDate", startDate.toISOString()); |     if (startDate) params = params.append("sDate", startDate.toISOString()); | ||||||
|     if (endDate) params = params.append("eDate", endDate.toISOString()); |     if (endDate) params = params.append("eDate", endDate.toISOString()); | ||||||
|     if(exactMatchStartOrEndDate !== undefined) params = params.append("exactMatchStartOrEndDate", exactMatchStartOrEndDate);   |     if(exactMatchStartOrEndDate !== undefined) params = params.append("exactMatchStartOrEndDate", exactMatchStartOrEndDate); | ||||||
|      |     if(owner) params = params.append("owner", owner); | ||||||
|  |  | ||||||
|     if(skip) params = params.append("skip", skip); |     if(skip) params = params.append("skip", skip); | ||||||
|     if(take) params = params.append("take", take); |     if(take) params = params.append("take", take); | ||||||
|  |     if(indexed) params = params.append("ind",indexed?"true":"false"); | ||||||
|  |     if(atItemLocationItemCode) params = params.append("ail",atItemLocationItemCode); | ||||||
|     return this.httpClient.get<IItem[]>(`${this.ApiEndpoint()}/api/v1/items/${parentcode}/children`, { params: params }); |     return this.httpClient.get<IItem[]>(`${this.ApiEndpoint()}/api/v1/items/${parentcode}/children`, { params: params }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   getChildItemListCount(parentcode: string, itemType?: string,dataFilter?: any, |   getChildItemListCount(parentcode: string, itemType?: string,dataFilter?: any, | ||||||
|                         startDate?: Date, endDate?: Date, exactMatchStartOrEndDate?: boolean): Observable<number> { |                         startDate?: Date, endDate?: Date, exactMatchStartOrEndDate?: boolean, owner?:string, | ||||||
|  |                         indexed?: boolean): Observable<number> { | ||||||
|     let params = new HttpParams(); |     let params = new HttpParams(); | ||||||
|     if(itemType != null) { |     if(itemType != null) { | ||||||
|       params = params.append("it", itemType); |       params = params.append("it", itemType); | ||||||
| @@ -145,7 +151,9 @@ export class ItemService { | |||||||
|     } |     } | ||||||
|     if (startDate) params = params.append("sDate", startDate.toISOString()); |     if (startDate) params = params.append("sDate", startDate.toISOString()); | ||||||
|     if (endDate) params = params.append("eDate", endDate.toISOString()); |     if (endDate) params = params.append("eDate", endDate.toISOString()); | ||||||
|     if(exactMatchStartOrEndDate !== undefined) params = params.append("exactMatchStartOrEndDate", exactMatchStartOrEndDate);   |     if(exactMatchStartOrEndDate !== undefined) params = params.append("exactMatchStartOrEndDate", exactMatchStartOrEndDate); | ||||||
|  |     if(owner) params = params.append("owner", owner); | ||||||
|  |     if(indexed) params = params.append("ind",indexed?"true":"false"); | ||||||
|     return this.httpClient.get<number>(`${this.ApiEndpoint()}/api/v1/items/${parentcode}/children/count`, { params: params }); |     return this.httpClient.get<number>(`${this.ApiEndpoint()}/api/v1/items/${parentcode}/children/count`, { params: params }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -15,37 +15,37 @@ export class ItemTypeService { | |||||||
|  |  | ||||||
|      getIcon(itemType: string) { |      getIcon(itemType: string) { | ||||||
|        let icon = "fal fa-file"; |        let icon = "fal fa-file"; | ||||||
|        if (this.itemTypes[itemType]) icon = this.itemTypes[itemType].icon; |        if (this.itemTypes && this.itemTypes[itemType]) icon = this.itemTypes[itemType].icon; | ||||||
|        return icon; |        return icon; | ||||||
|      } |      } | ||||||
|  |  | ||||||
|      getColor(itemType: string) { |      getColor(itemType: string) { | ||||||
|        let color = "#000000"; |        let color = "#000000"; | ||||||
|        if (this.itemTypes[itemType]) color = this.itemTypes[itemType].iconColor; |        if (this.itemTypes && this.itemTypes[itemType]) color = this.itemTypes[itemType].iconColor; | ||||||
|        return color; |        return color; | ||||||
|      } |      } | ||||||
|  |  | ||||||
|      getExtraAttributes(itemType: string) { |      getExtraAttributes(itemType: string) { | ||||||
|       let extraAttributes = null; |       let extraAttributes = null; | ||||||
|       if (this.itemTypes[itemType] && this.itemTypes[itemType]?.extraAttributes) extraAttributes = this.itemTypes[itemType]?.extraAttributes; |       if (this.itemTypes && this.itemTypes[itemType] && this.itemTypes[itemType]?.extraAttributes) extraAttributes = this.itemTypes[itemType]?.extraAttributes; | ||||||
|       return extraAttributes; |       return extraAttributes; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     getSchema(itemType: string): string { |     getSchema(itemType: string): string { | ||||||
|       let schema = null; |       let schema = null; | ||||||
|       if (this.itemTypes[itemType]) schema = this.itemTypes[itemType].schema; |       if (this.itemTypes && this.itemTypes[itemType]) schema = this.itemTypes[itemType].schema; | ||||||
|       return schema; |       return schema; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|      hasViewer(item: IItem) { |      hasViewer(item: IItem) { | ||||||
|         const itemType: string = item.itemType; |         const itemType: string = item.itemType; | ||||||
|         if (this.itemTypes[itemType]) return this.itemTypes[itemType].viewer !== undefined; |         if (this.itemTypes && this.itemTypes[itemType]) return this.itemTypes[itemType].viewer !== undefined; | ||||||
|         return false; |         return false; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       hasEditor(item: IItem) { |       hasEditor(item: IItem) { | ||||||
|         const itemType: string = item.itemType; |         const itemType: string = item.itemType; | ||||||
|         if (this.itemTypes[itemType]) return this.itemTypes[itemType].editor !== undefined; |         if (this.itemTypes && this.itemTypes[itemType]) return this.itemTypes[itemType].editor !== undefined; | ||||||
|         return false; |         return false; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										40
									
								
								projects/common/src/fm/services/task.service.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								projects/common/src/fm/services/task.service.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | import {Injectable} from '@angular/core'; | ||||||
|  | import {HttpClient, HttpParams} from '@angular/common/http'; | ||||||
|  | import {AppConfig} from '../shared/app.config'; | ||||||
|  |  | ||||||
|  | @Injectable({ | ||||||
|  |   providedIn: 'root', | ||||||
|  | }) | ||||||
|  | export class TaskService { | ||||||
|  |   constructor(public httpClient: HttpClient, public appConfig: AppConfig) { | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   ApiEndpoint() { | ||||||
|  |     return this.appConfig.getConfig("apiEndPoint"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   getTask(taskCode: string) { | ||||||
|  |     return this.httpClient.get<any>(`${this.appConfig.getConfig('apiEndPoint')}/api/v1/tasks/${taskCode}`) | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   getTasks(workflowCode: string, itemCode: string = null, taskType: string = null, skip = 0, take = 25) { | ||||||
|  |     let params = new HttpParams(); | ||||||
|  |  | ||||||
|  |     if (workflowCode) { | ||||||
|  |       params = params.append('workflowCode', workflowCode); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (itemCode) { | ||||||
|  |       params = params.append('itemCode', itemCode); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (taskType) { | ||||||
|  |       params = params.append('taskType', taskType); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     params = params.append('skip', skip.toString()); | ||||||
|  |     params = params.append('take', take.toString()); | ||||||
|  |     return this.httpClient.get<any>(`${this.appConfig.getConfig('apiEndPoint')}/api/v1/tasks`, | ||||||
|  |       {params: params}) | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -31,8 +31,8 @@ | |||||||
|     "tslib": "^2.3.0" |     "tslib": "^2.3.0" | ||||||
|   }, |   }, | ||||||
|   "peerDependencies": { |   "peerDependencies": { | ||||||
|     "@angular/common": "^18.2.3", |     "@angular/common": "18.2.3", | ||||||
|     "@angular/core": "^18.2.3", |     "@angular/core": "18.2.3", | ||||||
|     "ol": "^8.2.0" |     "ol": "^8.2.0" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user