diff --git a/spec/karma.conf2.js b/spec/karma.conf2.js deleted file mode 100644 index 6838075963cc65dabb8447d9347abd9d3973b06c..0000000000000000000000000000000000000000 --- a/spec/karma.conf2.js +++ /dev/null @@ -1,69 +0,0 @@ -// Karma configuration -// Generated on Mon Aug 06 2018 16:37:20 GMT+0200 (CEST) - -module.exports = function(config) { - config.set({ - - // base path that will be used to resolve all patterns (eg. files, exclude) - basePath: '..', - - - // frameworks to use - // available frameworks: https://npmjs.org/browse/keyword/karma-adapter - frameworks: ['jasmine'], - - - // list of files / patterns to load in the browser - files: [ - './spec/test.ts' - ], - - - // list of files / patterns to exclude - exclude: [ - ], - - - // preprocess matching files before serving them to the browser - // available preprocessors: https://npmjs.org/browse/keyword/karma-preprocessor - preprocessors: { - }, - - - // test results reporter to use - // possible values: 'dots', 'progress' - // available reporters: https://npmjs.org/browse/keyword/karma-reporter - reporters: ['progress'], - - - // web server port - port: 9876, - - - // enable / disable colors in the output (reporters and logs) - colors: true, - - - // level of logging - // possible values: config.LOG_DISABLE || config.LOG_ERROR || config.LOG_WARN || config.LOG_INFO || config.LOG_DEBUG - logLevel: config.LOG_INFO, - - - // enable / disable watching file and executing tests whenever any file changes - autoWatch: true, - - - // start these browsers - // available browser launchers: https://npmjs.org/browse/keyword/karma-launcher - browsers: ['Chrome'], - - - // Continuous Integration mode - // if true, Karma captures browsers, runs the tests and exits - singleRun: false, - - // Concurrency level - // how many browser should be started simultaneous - concurrency: Infinity - }) -} diff --git a/src/atlasViewer/atlasViewer.apiService.service.ts b/src/atlasViewer/atlasViewer.apiService.service.ts index cdd6fb3c3f7317ff4f72990f5bb81ea4ff55ecdf..bd75872dd0b778520175cd11c922eab92c282689 100644 --- a/src/atlasViewer/atlasViewer.apiService.service.ts +++ b/src/atlasViewer/atlasViewer.apiService.service.ts @@ -1,8 +1,12 @@ import { Injectable } from "@angular/core"; import { Store, select } from "@ngrx/store"; -import { ViewerStateInterface, safeFilter, getLabelIndexMap } from "../services/stateStore.service"; +import { ViewerStateInterface, safeFilter, getLabelIndexMap, isDefined } from "../services/stateStore.service"; import { Observable } from "rxjs"; -import { map, distinctUntilChanged } from "rxjs/operators"; +import { map, distinctUntilChanged, filter } from "rxjs/operators"; +import { BsModalService } from "ngx-bootstrap/modal"; +import { ModalUnit } from "./modalUnit/modalUnit.component"; +import { ModalHandler } from "../util/pluginHandlerClasses/modalHandler"; +import { ToastHandler } from "../util/pluginHandlerClasses/toastHandler"; declare var window @@ -14,12 +18,15 @@ export class AtlasViewerAPIServices{ private loadedTemplates$ : Observable<any> private selectParcellation$ : Observable<any> + private selectTemplate$ : Observable<any> + private darktheme : boolean public interactiveViewer : InteractiveViewerInterface public loadedLibraries : Map<string,{counter:number,src:HTMLElement|null}> = new Map() constructor( - private store : Store<ViewerStateInterface> + private store : Store<ViewerStateInterface>, + private modalService: BsModalService ){ this.loadedTemplates$ = this.store.pipe( @@ -28,6 +35,13 @@ export class AtlasViewerAPIServices{ map(state=>state.fetchedTemplates) ) + this.selectTemplate$ = this.store.pipe( + select('viewerState'), + filter(state => isDefined(state) && isDefined(state.templateSelected)), + map(state => state.templateSelected), + distinctUntilChanged((t1, t2) => t1.name === t2.name) + ) + this.selectParcellation$ = this.store.pipe( select('viewerState'), safeFilter('parcellationSelected'), @@ -63,7 +77,37 @@ export class AtlasViewerAPIServices{ map(state=>state.fetchedDataEntries) ) }, - uiHandle : {}, + uiHandle : { + getModalHandler : () => { + const handler = new ModalHandler() + let modalRef + handler.show = () => { + modalRef = this.modalService.show(ModalUnit, { + initialState : { + title : handler.title, + body : handler.body + ? handler.body + : 'handler.body has yet been defined ...', + footer : handler.footer + }, + class : this.darktheme ? 'darktheme' : 'not-darktheme', + backdrop : handler.dismissable ? true : 'static', + keyboard : handler.dismissable + }) + } + handler.hide = () => { + if(modalRef){ + modalRef.hide() + modalRef = null + } + } + return handler + }, + /* to be overwritten by atlasViewer.component.ts */ + getToastHandler : () => { + throw new Error('getToast Handler not overwritten by atlasViewer.component.ts') + } + }, pluginControl : { loadExternalLibraries : ()=>Promise.reject('load External Library method not over written') , @@ -79,6 +123,7 @@ export class AtlasViewerAPIServices{ private init(){ this.loadedTemplates$.subscribe(templates=>this.interactiveViewer.metadata.loadedTemplates = templates) this.selectParcellation$.subscribe(parcellation => this.interactiveViewer.metadata.regionsLabelIndexMap = getLabelIndexMap(parcellation.regions)) + this.selectTemplate$.subscribe(template => this.darktheme = template.useTheme === 'dark') } } @@ -113,7 +158,8 @@ export interface InteractiveViewerInterface{ } uiHandle : { - + getModalHandler : () => ModalHandler + getToastHandler : () => ToastHandler } pluginControl : { @@ -123,6 +169,7 @@ export interface InteractiveViewerInterface{ } } + export interface NGLayerObj{ } \ No newline at end of file diff --git a/src/atlasViewer/atlasViewer.component.ts b/src/atlasViewer/atlasViewer.component.ts index 6b80c0f37c8417600a0bcfbc6b2aabbea30e80ba..a982e88385f8e7bad882306aa9df2cc97bffb04f 100644 --- a/src/atlasViewer/atlasViewer.component.ts +++ b/src/atlasViewer/atlasViewer.component.ts @@ -19,6 +19,7 @@ import { PluginServices } from "./atlasViewer.pluginService.service"; import '../res/css/extra_styles.css' import { NehubaContainer } from "../ui/nehubaContainer/nehubaContainer.component"; +import { ToastHandler } from "../util/pluginHandlerClasses/toastHandler"; @Component({ selector: 'atlas-viewer', @@ -105,6 +106,32 @@ export class AtlasViewer implements OnDestroy, OnInit, AfterViewInit { ngOnInit() { + this.apiService.interactiveViewer.uiHandle.getToastHandler = () => { + const handler = new ToastHandler() + let toastComponent:ComponentRef<ToastComponent> + handler.show = () => { + toastComponent = this.toastContainer.createComponent(this.toastComponentFactory) + + toastComponent.instance.dismissable = handler.dismissable + toastComponent.instance.message = handler.message + toastComponent.instance.timeout = handler.timeout + + const _subscription = toastComponent.instance.dismissed.subscribe(userInitiated => { + _subscription.unsubscribe() + handler.hide() + }) + } + + handler.hide = () => { + if(toastComponent){ + toastComponent.destroy() + toastComponent = null + } + } + + return handler + } + this.apiService.interactiveViewer.pluginControl.loadExternalLibraries = (libraries: string[]) => new Promise((resolve, reject) => { const srcHTMLElement = libraries.map(libraryName => ({ name: libraryName, diff --git a/src/atlasViewer/atlasViewer.template.html b/src/atlasViewer/atlasViewer.template.html index cd80d350841f899b564a3f97deb3aed34021db89..320501b42f7d440f5024019f3d3dcb78548241b9 100644 --- a/src/atlasViewer/atlasViewer.template.html +++ b/src/atlasViewer/atlasViewer.template.html @@ -41,7 +41,6 @@ </ng-template> <ng-template #pluginFactory> - </ng-template> </div> diff --git a/src/atlasViewer/modalUnit/modalUnit.component.ts b/src/atlasViewer/modalUnit/modalUnit.component.ts index 1bf7a1950438635b15df3160ae5f087c59284f1f..a7fd52717d9086a2fd63cb7451ec50d0846fc2cb 100644 --- a/src/atlasViewer/modalUnit/modalUnit.component.ts +++ b/src/atlasViewer/modalUnit/modalUnit.component.ts @@ -10,6 +10,7 @@ import { Component, Input, ViewContainerRef } from '@angular/core' export class ModalUnit{ @Input() title : string @Input() body : string = 'Modal Body Text' + @Input() footer: string constructor(public viewContainerRef : ViewContainerRef){ diff --git a/src/atlasViewer/modalUnit/modalUnit.style.css b/src/atlasViewer/modalUnit/modalUnit.style.css index e69de29bb2d1d6434b8b29ae775ad8c2e48c5391..619233185073476c359eb2b6188f1a5280a166c2 100644 --- a/src/atlasViewer/modalUnit/modalUnit.style.css +++ b/src/atlasViewer/modalUnit/modalUnit.style.css @@ -0,0 +1,8 @@ +:host-context(.darktheme) div.modal-header, +:host-context(.darktheme) div.modal-body, +:host-context(.darktheme) div.modal-footer +{ + background-color:rgba(50,50,50,1.0); + border-color: rgba(0,0,0,0.2); + color:white; +} \ No newline at end of file diff --git a/src/atlasViewer/modalUnit/modalUnit.template.html b/src/atlasViewer/modalUnit/modalUnit.template.html index 4caa1075bb0b55db1eef3a5428c5cd19c07fa91e..abb9fbb959d040759ca2f92942e951989180bb03 100644 --- a/src/atlasViewer/modalUnit/modalUnit.template.html +++ b/src/atlasViewer/modalUnit/modalUnit.template.html @@ -6,4 +6,8 @@ <div class = "modal-body"> {{ body }} +</div> + +<div *ngIf = "footer" class = "modal-footer"> + {{ footer }} </div> \ No newline at end of file diff --git a/src/components/dropdown/dropdown.animation.ts b/src/components/dropdown/dropdown.animation.ts new file mode 100644 index 0000000000000000000000000000000000000000..9fe53cea634fa63c9b4927eaca3d5c11bd0b6569 --- /dev/null +++ b/src/components/dropdown/dropdown.animation.ts @@ -0,0 +1,21 @@ +import { trigger, state, style, transition, animate } from "@angular/animations"; + + +export const dropdownAnimation = trigger('showState',[ + state('show', + style({ + opacity : '1.0' + }) + ), + state('hide', + style({ + opacity : '0.0' + }) + ), + transition('show => hide', [ + animate('150ms ease-in') + ]), + transition('hide => show',[ + animate('150ms ease-out') + ]) +]) \ No newline at end of file diff --git a/src/components/dropdown/dropdown.component.ts b/src/components/dropdown/dropdown.component.ts index c13104ffc81e5335ac2c73c7fdbcbda7f6067f81..5495127870b18f2286365e6b36ef622e7b85e56b 100644 --- a/src/components/dropdown/dropdown.component.ts +++ b/src/components/dropdown/dropdown.component.ts @@ -1,4 +1,5 @@ import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy } from "@angular/core"; +import { dropdownAnimation } from "./dropdown.animation"; @Component({ selector : 'dropdown', @@ -22,6 +23,9 @@ ul > li:not(.selected) > span:before } ` ], + animations:[ + dropdownAnimation + ], changeDetection : ChangeDetectionStrategy.OnPush }) @@ -34,4 +38,9 @@ export class DropdownComponent{ @Input() activeDisplay : (obj:any|null)=>string = (obj)=>obj ? obj.name : `Please select an item.` @Output() itemSelected : EventEmitter<any> = new EventEmitter() + + open : boolean = false + isOpenChange(event:any){ + console.log(event) + } } \ No newline at end of file diff --git a/src/components/toast/toast.component.ts b/src/components/toast/toast.component.ts index c12eeb7994f4330acb8ae9444c1a19450e230136..f0ed8c9b8396b06366d7e26b9b7a2e934ac42178 100644 --- a/src/components/toast/toast.component.ts +++ b/src/components/toast/toast.component.ts @@ -1,4 +1,4 @@ -import { Component, Input, ViewContainerRef, ViewChild } from "@angular/core"; +import { Component, Input, ViewContainerRef, ViewChild, Output, EventEmitter } from "@angular/core"; @Component({ @@ -8,20 +8,30 @@ import { Component, Input, ViewContainerRef, ViewChild } from "@angular/core"; }) export class ToastComponent{ - @Input() message : string = 'template' + @Input() message : string @Input() timeout : number = 0 @Input() dismissable : boolean = true - @ViewChild('message',{read:ViewContainerRef}) messageContainer : ViewContainerRef - constructor(){ + @Output() dismissed : EventEmitter<boolean> = new EventEmitter() + + private timeoutId : number + @ViewChild('messageContainer',{read:ViewContainerRef}) messageContainer : ViewContainerRef + constructor(){ + + } + + + ngOnInit(){ + if(this.timeout > 0) this.timeoutId = setTimeout(() => this.dismissed.emit(false), this.timeout) } dismiss(event:MouseEvent){ event.preventDefault() event.stopPropagation() - - console.warn('dismiss') + if(this.timeoutId) clearTimeout(this.timeoutId) + + this.dismissed.emit(true) } } \ No newline at end of file diff --git a/src/components/toast/toast.template.html b/src/components/toast/toast.template.html index 9d4426e628819ba2f7ade1b7629d70aa4f42eea2..579970e3fea5ade79bf5594d500abb5d75db1e1a 100644 --- a/src/components/toast/toast.template.html +++ b/src/components/toast/toast.template.html @@ -1,9 +1,12 @@ <div container> <div message> - <ng-template #message> + <ng-template #messageContainer> </ng-template> </div> + <div *ngIf = "message"> + {{ message }} + </div> <div (click) = "dismiss($event)" *ngIf = "dismissable" close> <i class = "glyphicon glyphicon-remove"></i> </div> diff --git a/src/layouts/mainside/mainside.style.css b/src/layouts/mainside/mainside.style.css index 9f4be14ea9d4da5428fe100796858530446dee82..06af49b9ae31a8cd42ecbfd0d7a948eb33a96d7a 100644 --- a/src/layouts/mainside/mainside.style.css +++ b/src/layouts/mainside/mainside.style.css @@ -24,18 +24,18 @@ div[resizeSliver] :host-context([darktheme="false"]) div[resizeSliver] { - background-color:rgba(30,30,30,0.08); + background-color:rgba(230,230,230,0.8); } :host-context([darktheme="true"]) div[resizeSliver] { - background-color:rgba(245,245,245,0.3); + background-color:rgba(50,50,50,0.8); color:rgba(245,245,245,0.8); } div[resizeSliver] { - z-index: 3; + z-index: 5; display:flex; flex-direction: row; align-items: center; diff --git a/src/plugin_examples/newWebJugex/script.js b/src/plugin_examples/newWebJugex/script.js index 407425c4d5a4de74f390c1955cf8b4e56d60b17c..1a0652a0c245b5cea2aedac2f27548e10a4240fc 100644 --- a/src/plugin_examples/newWebJugex/script.js +++ b/src/plugin_examples/newWebJugex/script.js @@ -1,5 +1,10 @@ (() => { + const handler = interactiveViewer.uiHandle.getToastHandler() + handler.message = 'hohoho' + handler.dismissable = true + handler.show() + const backendRoot = `http://medpc055.ime.kfa-juelich.de:8005` const srcRoot = 'http://localhost:10080/newWebJugex' @@ -75,20 +80,29 @@ removeGene: function (gene) { this.chosengenes = this.chosengenes.filter(g => g !== gene) }, + exportGene: function(){ + this.$refs.exportGeneAnchor.setAttribute('download', `${Date.now()}.csv`) + this.$refs.exportGeneAnchor.click() + console.log('export gene') + }, startAnalysis: function () { const body = { area1: { name: this.roi1, PMapURL: this.regionNameToPMapURLMap.get(this.roi1) }, - mode: false, area2: { name: this.roi2, PMapURL: this.regionNameToPMapURLMap.get(this.roi2) }, - hemisphere: 'left', - selectedGenes: this.chosengenes, - threshold: '0.2' + + simpleMode: this.simpleMode, + singleProbeMode: this.singleProbeMode, + ignoreCustomProbe: this.ignoreCustomProbe, + + lh: this.lefthemisphere, + rh: this.righthemisphere, + selectedGenes: this.chosengenes } this.$emit('start-analysis', Object.assign({}, body)) @@ -101,6 +115,9 @@ placeholderTextRoi2: function () { return this.roi2 === '' ? 'Start typing to search ...': this.roi2 }, + chosenGeneCommaJoined: function(){ + return this.chosengenes.join(',') + }, getAllgenes: function () { return this.allgenes } @@ -182,29 +199,33 @@ coorddata : null }), computed: { + filename: function(){ + return `pval_nPerm${this.querydata.nPermutations}_${this.querydata.singleProbeMode ? 'singleProbeMode' : 'allProbeMode'}` + }, dateString: function () { return `${this.datenow.getFullYear().toString()}${normaliseToTwoDigit(this.datenow.getMonth() + 1)}${normaliseToTwoDigit(this.datenow.getDate())}_${normaliseToTwoDigit(this.datenow.getHours())}${normaliseToTwoDigit(this.datenow.getMinutes())}` } }, mounted: function () { - fetch(`${backendRoot}/jugex`,{ - method: 'POST', - headers: { - 'Content-Type': 'application/json' - }, - body: JSON.stringify(this.querydata) - }) - .then(res=>res.json()) - .then(json => { - this.status = true - const rjson = jugexResponseParser(json) - this.pvaldata = rjson.pval - this.coorddata = rjson.coord - }) - .catch(e => { - this.status = true - this.error = JSON.stringify(e) - }) + console.log(this.querydata) + // fetch(`${backendRoot}/jugex`,{ + // method: 'POST', + // headers: { + // 'Content-Type': 'application/json' + // }, + // body: JSON.stringify(this.querydata) + // }) + // .then(res=>res.json()) + // .then(json => { + // this.status = true + // const rjson = jugexResponseParser(json) + // this.pvaldata = rjson.pval + // this.coorddata = rjson.coord + // }) + // .catch(e => { + // this.status = true + // this.error = JSON.stringify(e) + // }) } }) @@ -224,6 +245,7 @@ }) controller.$on('start-analysis', (data) => { + /* validation (?) */ result.addAnalysis(data) }) }) diff --git a/src/plugin_examples/newWebJugex/template.html b/src/plugin_examples/newWebJugex/template.html index 615a880186622367a5521af9de8a306c53ec7557..657b00eb889f1347b10b2ed8611853552f2317fe 100644 --- a/src/plugin_examples/newWebJugex/template.html +++ b/src/plugin_examples/newWebJugex/template.html @@ -82,8 +82,12 @@ </div> </div> <div class = "input-group-btn"> - <div webjugex-tooltip = "saves the gene list as a comma delimited, utf8 encoded csv file" class = "btn btn-default fzj.xg.newWebJugex.geneBtns"> + <div + @click = "exportGene" + webjugex-tooltip = "saves the gene list as a comma delimited, utf8 encoded csv file" + class = "btn btn-default fzj.xg.newWebJugex.geneBtns"> Export + <a style = "display:none" ref = "exportGeneAnchor" :href = "'data:text/csv;charset=utf-8,' + chosenGeneCommaJoined">button</a> </div> </div> diff --git a/src/plugin_examples/plugin_api.md b/src/plugin_examples/plugin_api.md index bbf95aecadfe3fdf38ba8ef446b1a866c710d5db..f68bcbf4a7cc63b3c082eda35cad2ccdd76a2205 100644 --- a/src/plugin_examples/plugin_api.md +++ b/src/plugin_examples/plugin_api.md @@ -102,23 +102,16 @@ window.interactiveViewer - modalControl - - *getModalHandler()* returns a modalHandlerObject - - - *modalHandler* + - *getModalHandler()* returns a modalHandlerObject, which has the following methods/properties: - *hide()* : Dynamically hides the modal - *show()* : Shows the modal - - *onHide(callback(reason)=>void)* : Attaches an onHide callback. - - *onHidden(callback(reason)=>void)* : Attaches an onHidden callback. - - *onShow(callback(reason)=>void)* : Attaches an onShow callback. - - *onShown(callback(reason)=>void)* : Attaches an onShown callback. - title : title of the modal (String) - - body : body of the modal shown (JSON, Array, String) + - body : body of the modal shown (String) - footer : footer of the modal (String) - - config : config of the modal - - - *viewingModeBSubject* BehaviourSubject emitting strings when viewing mode has changed. + - dismissable : whether the modal is dismissable on click backdrop/esc key (Boolean) *n.b. if true, users will not be able to interact with the viewer unless you specifically call `handler.hide()`* + - toastControl - pluginControl - *loadExternalLibraries([LIBRARY_NAME_1,LIBRARY_NAME_2])* Function that loads external libraries. Pass the name of the libraries as an Array of string, and returns a Promise. When promise resolves, the libraries are loaded. **n.b.** while unlikely, there is a possibility that multiple requests to load external libraries in quick succession can cause the promise to resolve before the library is actually loaded. diff --git a/src/ui/databrowser/databrowser.component.ts b/src/ui/databrowser/databrowser.component.ts index 96f6e9d06ce55300958ff4328180a20b0024f82e..c39e3d47e8ee3efa14b1aff92bab1c1176694330 100644 --- a/src/ui/databrowser/databrowser.component.ts +++ b/src/ui/databrowser/databrowser.component.ts @@ -34,7 +34,7 @@ export class DataBrowserUI implements OnDestroy,OnInit{ spatialTotalNo : number = 0 hideDataTypes : Set<string> = new Set() - private _spatialDataVisible : boolean = true + private _spatialDataVisible : boolean = false private spatialSearchObj : {center:[number,number,number],searchWidth:number,templateSpace : string,pageNo:number} dedicatedViewString : string | null @@ -213,6 +213,9 @@ export class DataBrowserUI implements OnDestroy,OnInit{ if(isDefined(state.spatialDataVisible)) this._spatialDataVisible = state.spatialDataVisible + if(this._spatialDataVisible === false) + return + if(this.spatialPagination === this.spatialSearchObj.pageNo) return diff --git a/src/util/pipes/copyProperty.pipe.spec.ts b/src/util/pipes/copyProperty.pipe.spec.ts index 993af71f0a788a34c425b290555d825c32904305..548df650ab385b60a615859327d9c4b8605c82ca 100644 --- a/src/util/pipes/copyProperty.pipe.spec.ts +++ b/src/util/pipes/copyProperty.pipe.spec.ts @@ -1,19 +1,27 @@ import { CopyPropertyPipe } from './copyProperty.pipe' import {} from 'jasmine' +const array = [{ + name : 'name1', + key1 : 'value1' +},{ + name : 'name2', + key1 : 'value2' +},{ + name : 'name3', + key1 : 'value3', + key2 : 'oldvalue3' +},{ + name : 'name4', + key2 : 'oldValue4' +}] + describe('copyProperty.pipe works as expected', () => { it('jasmine works', () => { expect(1).toBe(1) }) - it('pipe should work', () => { + it('copyProperty pipe should copy value of key1 as value of key2, even means overwriting old value', () => { const pipe = new CopyPropertyPipe() - const array = [{ - name : 'name1', - key1 : 'value1' - },{ - name : 'name2', - key1 : 'value2' - }] const newItem = pipe.transform(array,'key1','key2') expect(newItem[0]).toEqual({ @@ -27,5 +35,32 @@ describe('copyProperty.pipe works as expected', () => { key1 : 'value2', key2 : 'value2' }) + + expect(newItem[2]).toEqual({ + name : 'name3', + key1 : 'value3', + key2 : 'value3' + }) + + expect(newItem[3]).toEqual({ + name : 'name4', + key2 : undefined + }) + }) + it('if given undefined or null as array input, will return an emtpy array', () => { + const pipe = new CopyPropertyPipe() + const nullItem = pipe.transform(null,'key1','key2') + expect(nullItem).toEqual([]) + + const undefinedItem = pipe.transform(undefined, 'key1','key2') + expect(undefinedItem).toEqual([]) + }) + it('if either keys are undefined, return the original array, or emtpy array if original array is undefined', () => { + const pipe = new CopyPropertyPipe() + const nokey1 = pipe.transform(array, null, 'key2') + expect(nokey1).toEqual(array) + + const nokey2 = pipe.transform(array, 'key1', null) + expect(nokey2).toEqual(array) }) }) \ No newline at end of file diff --git a/src/util/pipes/copyProperty.pipe.ts b/src/util/pipes/copyProperty.pipe.ts index e065c0943b360fe68ba197c4372919a92a90f315..0d9aec8897794536271d882d872b303ae577cfcb 100644 --- a/src/util/pipes/copyProperty.pipe.ts +++ b/src/util/pipes/copyProperty.pipe.ts @@ -5,9 +5,17 @@ import { PipeTransform, Pipe } from "@angular/core"; }) export class CopyPropertyPipe implements PipeTransform{ + private isDefined(obj){ + return typeof obj !== 'undefined' && obj !== null + } public transform(inputArray:any[],src:string,dest:string):any[]{ - if(!inputArray) + if(!this.isDefined(inputArray)) return [] + if(!this.isDefined(src) || !this.isDefined(dest) ) + return this.isDefined(inputArray) + ? inputArray + : [] + return inputArray.map(item=>{ const newObj = Object.assign({},item) newObj[dest] = item[src] diff --git a/src/util/pipes/groupDataEntriesByRegion.pipe.ts b/src/util/pipes/groupDataEntriesByRegion.pipe.ts index 3d2ab267de27c1c0d09126929543229469c40e76..7c2986db80ad5ebb4152f332f70aa402473cf3be 100644 --- a/src/util/pipes/groupDataEntriesByRegion.pipe.ts +++ b/src/util/pipes/groupDataEntriesByRegion.pipe.ts @@ -5,44 +5,38 @@ import { DataEntry } from "../../services/stateStore.service"; name : 'groupDatasetByRegion' }) export class GroupDatasetByRegion implements PipeTransform{ - public transform( - datasets:DataEntry[], - regions:any[] - ) : {region:any|null,searchResults:DataEntry[]}[] + public transform(datasets:DataEntry[], regions:any[]): {region:any|null,searchResults:DataEntry[]}[] { return datasets.reduce((acc,curr)=>{ - return (curr.regionName && curr.regionName.length > 0) ? - curr.regionName.reduce((acc2,reName)=>{ - const idx = acc - .findIndex(it => it.region === null ? - reName.regionName === 'none' : - it.region.name === reName.regionName ) + return (curr.regionName && curr.regionName.length > 0) + ? curr.regionName.reduce((acc2,reName)=>{ + const idx = acc.findIndex(it => it.region === null + ? reName.regionName === 'none' + : it.region.name === reName.regionName ) - return idx >= 0 ? - acc2.map((v,i)=> i === idx ? Object.assign({},v,{searchResults : v.searchResults.concat(curr)}) : v ) : - acc2.concat({ - region : this.getRegionFromRegionName(reName.regionName, regions), - searchResults : [ curr ] + return idx >= 0 + ? acc2.map((v,i)=> i === idx + ? Object.assign({},v,{searchResults : v.searchResults.concat(curr)}) + : v ) + : acc2.concat({ + region : this.getRegionFromRegionName(reName.regionName, regions), + searchResults : [ curr ] + }) + },acc) + : acc.findIndex(it=>it.region==null) >= 0 + ? acc.map(it=>it.region === null + ? Object.assign({}, it, {searchResults: it.searchResults.concat(curr)}) + : it) + : acc.concat({ + region : null, + searchResults : [curr] }) - },acc) : - acc.findIndex(it=>it.region==null) >= 0 ? - acc.map(it=>it.region === null ? - Object.assign({},it,{ - searchResults:it.searchResults.concat(curr) - }) : - it) : - acc.concat({ - region : null, - searchResults : [curr] - }) - - },[] as {region:any|null,searchResults:DataEntry[]}[]) + }, [] as {region:any|null,searchResults:DataEntry[]}[]) } private getRegionFromRegionName(regionName:string,regions:any[]):any|null{ const idx = regions.findIndex(re=>re.name == regionName) return idx >= 0 ? regions[idx] : null } - } \ No newline at end of file diff --git a/src/util/pipes/pathToNestedChildren.pipe.spec.ts b/src/util/pipes/pathToNestedChildren.pipe.spec.ts index 79fad121e1d5da39433eb85359f4654621368e97..3810225a2cd8830bb1e1b2b0417377c79c304dd0 100644 --- a/src/util/pipes/pathToNestedChildren.pipe.spec.ts +++ b/src/util/pipes/pathToNestedChildren.pipe.spec.ts @@ -2,10 +2,105 @@ * path to nested children should successfully convert flat array to nested objects */ - import {} from 'jasmine' +import { PathToNestedChildren } from './pathToNestedChildren.pipe' - describe('path to nested children', () => { - it('jasmine works', () => { - expect(1).toBe(1) - }) - }) \ No newline at end of file +import {} from 'jasmine' + +const array1 = [{ + pizza : 'pineapple', + path : 'root1' +},{ + path: 'root2' +}] + +const expectedArray1 = [{ + pizza : 'pineapple', + path : 'root1', + children : [] +},{ + path : 'root2', + children : [] +}] + +const array2 = [{ + path : 'root' +},{ + path : 'root/dir' +}] +const expectedArray2 = [{ + path : 'root', + children : [{ + path : 'dir', + children : [] + }] +}] + +const array3 = [{ + name : 'eagle', + path : 'root1/dir1' +},{ + path : 'root1/dir2' +},{ + path : 'root2/dir3' +},{ + path : 'root2/dir4' +}] +const expectedArray3 = [{ + path : 'root1', + children : [{ + name : 'eagle', + path : 'dir1', + children : [] + },{ + path : 'dir2', + children : [] + }] +},{ + path :'root2', + children :[{ + path : 'dir3', + children : [] + },{ + path : 'dir4', + children : [] + }] +}] + +const array4 = [{ + path : 'root1/3\\/4' +}] + +const expectedArray4 = [{ + path : 'root1', + children : [{ + path : '3\\/4', + children : [] + }] +}] + +const pipe = new PathToNestedChildren() + +describe('path to nested children', () => { + it('jasmine works', () => { + expect(1).toBe(1) + }) + + it('transforms all single heirachy to a flat structure', () => { + const transformed = pipe.transform(array1) + expect(transformed).toEqual(expectedArray1) + }) + + it('transforms nested hierachy correctly', () => { + + const transformed2 = pipe.transform(array2) + expect(transformed2).toEqual(expectedArray2) + + const transformed3 = pipe.transform(array3) + expect(transformed3).toEqual(expectedArray3) + }) + + it('handles escapes characters fine', () => { + const transformed4 = pipe.transform(array4) + expect(transformed4).toEqual(expectedArray4) + }) +}) \ No newline at end of file diff --git a/src/util/pipes/temp.ts b/src/util/pipes/temp.ts deleted file mode 100644 index 560fd1b94c479e0d18490d82b6f55afddcc43094..0000000000000000000000000000000000000000 --- a/src/util/pipes/temp.ts +++ /dev/null @@ -1,3 +0,0 @@ -export class TempClass{ - -} \ No newline at end of file diff --git a/src/util/pipes/treeSearch.pipe.ts b/src/util/pipes/treeSearch.pipe.ts index 5f26c05c662b0caae9b8a0d2c5736a0aa0413f6b..35a31e8aaa17372a33a26a9f1d483f8e3cbb4a64 100644 --- a/src/util/pipes/treeSearch.pipe.ts +++ b/src/util/pipes/treeSearch.pipe.ts @@ -1,6 +1,5 @@ import { PipeTransform, Pipe } from "@angular/core"; - @Pipe({ name : 'treeSearch' }) diff --git a/src/util/pluginHandlerClasses/modalHandler.ts b/src/util/pluginHandlerClasses/modalHandler.ts new file mode 100644 index 0000000000000000000000000000000000000000..07cc76cf3f6c031243869333b6459dfb6d115faa --- /dev/null +++ b/src/util/pluginHandlerClasses/modalHandler.ts @@ -0,0 +1,14 @@ +export class ModalHandler{ + + hide : () => void + show : () => void + // onHide : (callback: () => void) => void + // onHidden : (callback : () => void) => void + // onShow : (callback : () => void) => void + // onShown : (callback : () => void) => void + title : string + body : string + footer : String + + dismissable: boolean = true +} \ No newline at end of file diff --git a/src/util/pluginHandlerClasses/toastHandler.ts b/src/util/pluginHandlerClasses/toastHandler.ts new file mode 100644 index 0000000000000000000000000000000000000000..deae6df7ca4eb39412cee1e5d0398a7ea95d4198 --- /dev/null +++ b/src/util/pluginHandlerClasses/toastHandler.ts @@ -0,0 +1,7 @@ +export class ToastHandler{ + message : string = 'handler.body' + timeout : number = 3000 + dismissable : boolean = true + show : () => void + hide : () => void +} \ No newline at end of file