diff --git a/common/constants.js b/common/constants.js index d94ab3a30d5d7c6fbafb2380aadc40a725e28fed..1849b731b946a42438e8a9be05c6233f2f85bf0a 100644 --- a/common/constants.js +++ b/common/constants.js @@ -23,6 +23,7 @@ // overlay/layout specific SELECT_ATLAS: 'Select a different atlas', CONTEXT_MENU: `Viewer context menu`, + TOGGLE_FRONTAL_OCTANT: `Toggle perspective view frontal octant`, ZOOM_IN: 'Zoom in', ZOOM_OUT: 'Zoom out', MAXIMISE_VIEW: 'Maximise this view', diff --git a/src/services/state/ngViewerState.store.helper.ts b/src/services/state/ngViewerState.store.helper.ts index 834c5d5d53d4168a08a65c348ffbcf5e51eb5fa0..2d0ca1500648a3654e7108cc7a22588055ac50ba 100644 --- a/src/services/state/ngViewerState.store.helper.ts +++ b/src/services/state/ngViewerState.store.helper.ts @@ -22,3 +22,21 @@ export { ngViewerActionSetPanelOrder, ngViewerActionForceShowSegment, } + +import { + ngViewerSelectorClearView, + ngViewerSelectorClearViewEntries, + ngViewerSelectorNehubaReady, + ngViewerSelectorOctantRemoval, + ngViewerSelectorPanelMode, + ngViewerSelectorPanelOrder, +} from './ngViewerState/selectors' + +export { + ngViewerSelectorClearView, + ngViewerSelectorClearViewEntries, + ngViewerSelectorNehubaReady, + ngViewerSelectorOctantRemoval, + ngViewerSelectorPanelMode, + ngViewerSelectorPanelOrder, +} \ No newline at end of file diff --git a/src/services/state/ngViewerState.store.ts b/src/services/state/ngViewerState.store.ts index 2f708a826b8479ec41699b9c2f3528fb78763a5f..bc3b7fb8ce0abfcf2c3c1799a9fccf28238b71e3 100644 --- a/src/services/state/ngViewerState.store.ts +++ b/src/services/state/ngViewerState.store.ts @@ -12,6 +12,7 @@ import { PureContantService } from 'src/util'; import { PANELS } from './ngViewerState.store.helper' import { ngViewerActionToggleMax, ngViewerActionClearView, ngViewerActionSetPanelOrder, ngViewerActionSwitchPanelMode, ngViewerActionForceShowSegment, ngViewerActionNehubaReady } from './ngViewerState/actions'; import { generalApplyState } from '../stateStore.helper'; +import { ngViewerSelectorPanelMode, ngViewerSelectorPanelOrder } from './ngViewerState/selectors'; export function mixNgLayers(oldLayers: INgLayerInterface[], newLayers: INgLayerInterface|INgLayerInterface[]): INgLayerInterface[] { if (newLayers instanceof Array) { @@ -233,14 +234,12 @@ export class NgViewerUseEffect implements OnDestroy { ) this.panelOrder$ = this.store$.pipe( - select('ngViewerState'), - select('panelOrder'), + select(ngViewerSelectorPanelOrder), distinctUntilChanged(), ) this.panelMode$ = this.store$.pipe( - select('ngViewerState'), - select('panelMode'), + select(ngViewerSelectorPanelMode), distinctUntilChanged(), ) @@ -258,10 +257,10 @@ export class NgViewerUseEffect implements OnDestroy { this.maximiseOrder$ = toggleMaxmimise$.pipe( withLatestFrom( - combineLatest( + combineLatest([ this.panelOrder$, this.panelMode$, - ), + ]), ), filter(([_action, [_panelOrder, panelMode]]) => panelMode !== PANELS.SINGLE_PANEL), map(([ action, [ oldPanelOrder ] ]) => { @@ -277,10 +276,10 @@ export class NgViewerUseEffect implements OnDestroy { this.unmaximiseOrder$ = toggleMaxmimise$.pipe( withLatestFrom( - combineLatest( + combineLatest([ this.panelOrder$, this.panelMode$, - ), + ]), ), scan((acc, curr) => { const [action, [panelOrders, panelMode]] = curr @@ -329,10 +328,10 @@ export class NgViewerUseEffect implements OnDestroy { }), ) - this.toggleMaximiseCycleMessage$ = combineLatest( + this.toggleMaximiseCycleMessage$ = combineLatest([ this.toggleMaximiseMode$, this.pureConstantService.useTouchUI$, - ).pipe( + ]).pipe( filter(([_, useMobileUI]) => !useMobileUI), map(([toggleMaximiseMode, _]) => toggleMaximiseMode), filter(({ payload }) => payload.panelMode && payload.panelMode === PANELS.SINGLE_PANEL), diff --git a/src/services/state/ngViewerState/selectors.ts b/src/services/state/ngViewerState/selectors.ts index 85b992c44c585e5d75b997ba69f8ea908a935275..db84dcdb944e8e9f3d07b80a8d02464a492d4c3b 100644 --- a/src/services/state/ngViewerState/selectors.ts +++ b/src/services/state/ngViewerState/selectors.ts @@ -15,3 +15,23 @@ export const ngViewerSelectorClearView = createSelector( ngViewerSelectorClearViewEntries, keys => keys.length > 0 ) + +export const ngViewerSelectorPanelOrder = createSelector( + state => state['ngViewerState'], + ngViewerState => ngViewerState.panelOrder +) + +export const ngViewerSelectorPanelMode = createSelector( + state => state['ngViewerState'], + ngViewerState => ngViewerState.panelMode +) + +export const ngViewerSelectorOctantRemoval = createSelector( + state => state['ngViewerState'], + ngViewerState => ngViewerState.octantRemoval +) + +export const ngViewerSelectorNehubaReady = createSelector( + state => state['ngViewerState'], + ngViewerState => ngViewerState.nehubaReady +) diff --git a/src/services/state/viewerState/selectors.ts b/src/services/state/viewerState/selectors.ts index d41d1e0aa31ef56d0fef90a9b8fbea6fa8f4f293..44c4c8db76f62baea9ad3a656d7fcad184ed7ad2 100644 --- a/src/services/state/viewerState/selectors.ts +++ b/src/services/state/viewerState/selectors.ts @@ -63,6 +63,11 @@ export const viewerStateStandAloneVolumes = createSelector( viewerState => viewerState['standaloneVolumes'] ) +export const viewerStateSelectorNavigation = createSelector( + state => state['viewerState'], + viewerState => viewerState['navigation'] +) + export const viewerStateGetOverlayingAdditionalParcellations = createSelector( state => state[viewerStateHelperStoreName], state => state['viewerState'], diff --git a/src/ui/config/config.component.ts b/src/ui/config/config.component.ts index 756c15d1a6e80fd4f49a58dc1282642b331bc07f..4127b3bf990e732f1fccd4d94b69a5a0306d18f2 100644 --- a/src/ui/config/config.component.ts +++ b/src/ui/config/config.component.ts @@ -2,7 +2,7 @@ import { Component, OnDestroy, OnInit } from '@angular/core' import { select, Store } from '@ngrx/store'; import { combineLatest, Observable, Subscription } from 'rxjs'; import { debounceTime, distinctUntilChanged, map, startWith } from 'rxjs/operators'; -import { NG_VIEWER_ACTION_TYPES, SUPPORTED_PANEL_MODES } from 'src/services/state/ngViewerState.store'; +import { SUPPORTED_PANEL_MODES } from 'src/services/state/ngViewerState.store'; import { ngViewerActionSetPanelOrder } from 'src/services/state/ngViewerState.store.helper'; import { VIEWER_CONFIG_ACTION_TYPES, StateInterface as ViewerConfiguration } from 'src/services/state/viewerConfig.store' import { IavRootStoreInterface } from 'src/services/stateStore.service'; @@ -11,6 +11,8 @@ import {MatSlideToggleChange} from "@angular/material/slide-toggle"; import {MatSliderChange} from "@angular/material/slider"; import { PureContantService } from 'src/util'; import { ngViewerActionSwitchPanelMode } from 'src/services/state/ngViewerState/actions'; +import { ngViewerSelectorPanelMode, ngViewerSelectorPanelOrder } from 'src/services/state/ngViewerState/selectors'; +import { viewerStateSelectorNavigation } from 'src/services/state/viewerState/selectors'; const GPU_TOOLTIP = `Higher GPU usage can cause crashes on lower end machines` const ANIMATION_TOOLTIP = `Animation can cause slowdowns in lower end machines` @@ -73,19 +75,16 @@ export class ConfigComponent implements OnInit, OnDestroy { ) this.panelMode$ = this.store.pipe( - select('ngViewerState'), - select('panelMode'), + select(ngViewerSelectorPanelMode), startWith(SUPPORTED_PANEL_MODES[0]), ) this.panelOrder$ = this.store.pipe( - select('ngViewerState'), - select('panelOrder'), + select(ngViewerSelectorPanelOrder), ) this.viewerObliqueRotated$ = this.store.pipe( - select('viewerState'), - select('navigation'), + select(viewerStateSelectorNavigation), map(navigation => (navigation && navigation.orientation) || [0, 0, 0, 1]), debounceTime(100), map(isIdentityQuat), @@ -93,12 +92,12 @@ export class ConfigComponent implements OnInit, OnDestroy { distinctUntilChanged(), ) - this.panelTexts$ = combineLatest( + this.panelTexts$ = combineLatest([ this.panelOrder$.pipe( map(string => string.split('').map(s => Number(s))), ), this.viewerObliqueRotated$, - ).pipe( + ]).pipe( map(([arr, isObliqueRotated]) => arr.map(idx => (isObliqueRotated ? OBLIQUE_ROOT_TEXT_ORDER : ROOT_TEXT_ORDER)[idx]) as [string, string, string, string]), startWith(ROOT_TEXT_ORDER), ) diff --git a/src/ui/config/currentLayout/currentLayout.component.ts b/src/ui/config/currentLayout/currentLayout.component.ts index 40e8e8c4943de2fdb9139dc21ee62de4856e7ed7..313f1cd93467cb08955bd92f2dc728568427bf62 100644 --- a/src/ui/config/currentLayout/currentLayout.component.ts +++ b/src/ui/config/currentLayout/currentLayout.component.ts @@ -3,6 +3,7 @@ import { select, Store } from "@ngrx/store"; import { Observable } from "rxjs"; import { startWith } from "rxjs/operators"; import { SUPPORTED_PANEL_MODES } from "src/services/state/ngViewerState.store"; +import { ngViewerSelectorPanelMode } from "src/services/state/ngViewerState/selectors"; import { IavRootStoreInterface } from "src/services/stateStore.service"; @Component({ @@ -22,8 +23,7 @@ export class CurrentLayout { private store$: Store<IavRootStoreInterface>, ) { this.panelMode$ = this.store$.pipe( - select('ngViewerState'), - select('panelMode'), + select(ngViewerSelectorPanelMode), startWith(SUPPORTED_PANEL_MODES[0]), ) } diff --git a/src/ui/databrowserModule/databrowser.service.ts b/src/ui/databrowserModule/databrowser.service.ts index 8d3828f0b5cf21b0c9105aeacddc0b1fd62667d9..875daac2488ed97b858083629eb13a6da0269d0b 100644 --- a/src/ui/databrowserModule/databrowser.service.ts +++ b/src/ui/databrowserModule/databrowser.service.ts @@ -18,6 +18,7 @@ import { FilterDataEntriesByRegion } from "./util/filterDataEntriesByRegion.pipe import { datastateActionToggleFav, datastateActionUnfavDataset, datastateActionFavDataset } from "src/services/state/dataState/actions"; import { getIdFromFullId } from 'common/util' +import { viewerStateSelectorNavigation } from "src/services/state/viewerState/selectors"; const noMethodDisplayName = 'No methods described' @@ -113,8 +114,7 @@ export class DatabrowserService implements OnDestroy { ) this.viewportBoundingBox$ = this.store.pipe( - select('viewerState'), - select('navigation'), + select(viewerStateSelectorNavigation), distinctUntilChanged(), debounceTime(SPATIAL_SEARCH_DEBOUNCE), filter(v => !!v && !!v.position && !!v.zoom), diff --git a/src/ui/nehubaContainer/maximisePanelButton/maximisePanelButton.component.ts b/src/ui/nehubaContainer/maximisePanelButton/maximisePanelButton.component.ts index 36b2d9e882bb5a22b939c55902d7ecf84871a012..b3e5745f92e2ea7e9dbb2d95cc2298375b3a49b3 100644 --- a/src/ui/nehubaContainer/maximisePanelButton/maximisePanelButton.component.ts +++ b/src/ui/nehubaContainer/maximisePanelButton/maximisePanelButton.component.ts @@ -4,6 +4,7 @@ import { Observable } from "rxjs"; import { distinctUntilChanged, map } from "rxjs/operators"; import { PANELS } from 'src/services/state/ngViewerState.store.helper' import { ARIA_LABELS } from 'common/constants' +import { ngViewerSelectorPanelMode, ngViewerSelectorPanelOrder } from "src/services/state/ngViewerState/selectors"; const { MAXIMISE_VIEW, @@ -34,14 +35,12 @@ export class MaximmisePanelButton { private store$: Store<any>, ) { this.panelMode$ = this.store$.pipe( - select('ngViewerState'), - select('panelMode'), + select(ngViewerSelectorPanelMode), distinctUntilChanged(), ) this.panelOrder$ = this.store$.pipe( - select('ngViewerState'), - select('panelOrder'), + select(ngViewerSelectorPanelOrder), distinctUntilChanged(), ) diff --git a/src/ui/nehubaContainer/nehubaContainer.component.spec.ts b/src/ui/nehubaContainer/nehubaContainer.component.spec.ts index f187f412fe5eff1019a1828eafc62e9752fe2a64..3e5017b97f3c820dd1637442e736a6505f4edf04 100644 --- a/src/ui/nehubaContainer/nehubaContainer.component.spec.ts +++ b/src/ui/nehubaContainer/nehubaContainer.component.spec.ts @@ -1,5 +1,5 @@ -import { CUSTOM_ELEMENTS_SCHEMA } from '@angular/core' -import { async, TestBed } from "@angular/core/testing" +import { CUSTOM_ELEMENTS_SCHEMA, DebugElement } from '@angular/core' +import { TestBed, async, ComponentFixture, fakeAsync, tick, flush, discardPeriodicTasks } from "@angular/core/testing" import { NehubaContainer } from "./nehubaContainer.component" import { provideMockStore, MockStore } from "@ngrx/store/testing" import { defaultRootState } from 'src/services/stateStore.service' @@ -39,11 +39,16 @@ import { RegionAccordionTooltipTextPipe } from '../util' import { hot } from 'jasmine-marbles' import { of } from 'rxjs' import { HttpClientTestingModule, HttpTestingController } from '@angular/common/http/testing' +import { ngViewerSelectorPanelMode, ngViewerSelectorPanelOrder } from 'src/services/state/ngViewerState/selectors' +import { PANELS } from 'src/services/state/ngViewerState/constants' const { TOGGLE_SIDE_PANEL, EXPAND, - COLLAPSE + COLLAPSE, + ZOOM_IN, + ZOOM_OUT, + TOGGLE_FRONTAL_OCTANT } = ARIA_LABELS const _bigbrainJson = require('!json-loader!src/res/ext/bigbrain.json') @@ -425,5 +430,211 @@ describe('> nehubaContainer.component.ts', () => { it('> if something (region features/connectivity) exists, placeh holder text should be hdiden') }) }) + + describe('> panelCtrl', () => { + let fixture: ComponentFixture<NehubaContainer> + const setViewerLoaded = () => { + fixture.componentInstance.viewerLoaded = true + } + const ctrlElementIsVisible = (el: DebugElement) => { + const visible = (el.nativeElement as HTMLElement).getAttribute('data-viewer-controller-visible') + return visible === 'true' + } + beforeEach(() => { + fixture = TestBed.createComponent(NehubaContainer) + }) + it('> on start, all four ctrl panels exists', () => { + fixture.detectChanges() + setViewerLoaded() + fixture.detectChanges() + for (const idx of [0, 1, 2, 3]) { + const el = fixture.debugElement.query( + By.css(`[data-viewer-controller-index="${idx}"]`) + ) + expect(el).toBeTruthy() + } + }) + + it('> on start all four ctrl panels are invisible', () => { + + fixture.detectChanges() + setViewerLoaded() + fixture.detectChanges() + for (const idx of [0, 1, 2, 3]) { + const el = fixture.debugElement.query( + By.css(`[data-viewer-controller-index="${idx}"]`) + ) + expect(ctrlElementIsVisible(el)).toBeFalsy() + } + }) + + describe('> on hover, only the hovered panel have ctrl shown', () => { + + for (const idx of [0, 1, 2, 3]) { + + it(`> on hoveredPanelIndices$ emit ${idx}, the panel index ${idx} ctrl becomes visible`, fakeAsync(() => { + fixture.detectChanges() + const findPanelIndexSpy = spyOn<any>(fixture.componentInstance, 'findPanelIndex').and.callFake(() => { + return idx + }) + setViewerLoaded() + fixture.detectChanges() + const nativeElement = fixture.componentInstance['elementRef'].nativeElement + nativeElement.dispatchEvent(new MouseEvent('mouseover', { bubbles: true })) + + /** + * assert findPanelIndex called with event.target, i.e. native element in thsi case + */ + expect(findPanelIndexSpy).toHaveBeenCalledWith(nativeElement) + tick(200) + fixture.detectChanges() + + /** + * every panel index should be non visible + * only when idx matches, it can be visible + * n.b. this does not test visual visibility (which is controlled by extra-style.css) + * (which is also affected by global [ismobile] configuration) + * + * this merely test the unit logic, and sets the flag appropriately + */ + for (const iterativeIdx of [0, 1, 2, 3]) { + const el = fixture.debugElement.query( + By.css(`[data-viewer-controller-index="${iterativeIdx}"]`) + ) + if (iterativeIdx === idx) { + expect(ctrlElementIsVisible(el)).toBeTruthy() + } else { + expect(ctrlElementIsVisible(el)).toBeFalsy() + } + } + discardPeriodicTasks() + })) + } + + }) + + describe('> on maximise top right slice panel (idx 1)', () => { + beforeEach(() => { + const mockStore = TestBed.inject(MockStore) + mockStore.overrideSelector(ngViewerSelectorPanelMode, PANELS.SINGLE_PANEL) + mockStore.overrideSelector(ngViewerSelectorPanelOrder, '1230') + + fixture.detectChanges() + setViewerLoaded() + fixture.detectChanges() + }) + it('> toggle front octant btn not visible', () => { + + const toggleBtn = fixture.debugElement.query( + By.css(`[cell-i] [aria-label="${TOGGLE_FRONTAL_OCTANT}"]`) + ) + expect(toggleBtn).toBeFalsy() + }) + + it('> zoom in and out btns are visible', () => { + + const zoomInBtn = fixture.debugElement.query( + By.css(`[cell-i] [aria-label="${ZOOM_IN}"]`) + ) + + const zoomOutBtn = fixture.debugElement.query( + By.css(`[cell-i] [aria-label="${ZOOM_OUT}"]`) + ) + + expect(zoomInBtn).toBeTruthy() + expect(zoomOutBtn).toBeTruthy() + }) + + it('> zoom in btn calls fn with right param', () => { + const zoomViewSpy = spyOn(fixture.componentInstance, 'zoomNgView') + + const zoomInBtn = fixture.debugElement.query( + By.css(`[cell-i] [aria-label="${ZOOM_IN}"]`) + ) + zoomInBtn.triggerEventHandler('click', null) + expect(zoomViewSpy).toHaveBeenCalled() + const { args } = zoomViewSpy.calls.first() + expect(args[0]).toEqual(1) + /** + * zoom in < 1 + */ + expect(args[1]).toBeLessThan(1) + }) + it('> zoom out btn calls fn with right param', () => { + const zoomViewSpy = spyOn(fixture.componentInstance, 'zoomNgView') + + const zoomOutBtn = fixture.debugElement.query( + By.css(`[cell-i] [aria-label="${ZOOM_OUT}"]`) + ) + zoomOutBtn.triggerEventHandler('click', null) + expect(zoomViewSpy).toHaveBeenCalled() + const { args } = zoomViewSpy.calls.first() + expect(args[0]).toEqual(1) + /** + * zoom out > 1 + */ + expect(args[1]).toBeGreaterThan(1) + }) + }) + describe('> on maximise perspective panel', () => { + beforeEach(() => { + const mockStore = TestBed.inject(MockStore) + mockStore.overrideSelector(ngViewerSelectorPanelMode, PANELS.SINGLE_PANEL) + mockStore.overrideSelector(ngViewerSelectorPanelOrder, '3012') + + fixture.detectChanges() + setViewerLoaded() + fixture.detectChanges() + }) + it('> toggle octant btn visible and functional', () => { + const setOctantRemovalSpy = spyOn(fixture.componentInstance, 'setOctantRemoval') + + const toggleBtn = fixture.debugElement.query( + By.css(`[cell-i] [aria-label="${TOGGLE_FRONTAL_OCTANT}"]`) + ) + expect(toggleBtn).toBeTruthy() + toggleBtn.nativeElement.dispatchEvent( + new MouseEvent('click', { bubbles: true }) + ) + expect(setOctantRemovalSpy).toHaveBeenCalled() + }) + + it('> zoom in btn visible and functional', () => { + const zoomViewSpy = spyOn(fixture.componentInstance, 'zoomNgView') + + const zoomInBtn = fixture.debugElement.query( + By.css(`[cell-i] [aria-label="${ZOOM_IN}"]`) + ) + expect(zoomInBtn).toBeTruthy() + + zoomInBtn.triggerEventHandler('click', null) + expect(zoomViewSpy).toHaveBeenCalled() + const { args } = zoomViewSpy.calls.first() + expect(args[0]).toEqual(3) + /** + * zoom in < 1 + */ + expect(args[1]).toBeLessThan(1) + }) + it('> zoom out btn visible and functional', () => { + const zoomViewSpy = spyOn(fixture.componentInstance, 'zoomNgView') + + const zoomOutBtn = fixture.debugElement.query( + By.css(`[cell-i] [aria-label="${ZOOM_OUT}"]`) + ) + expect(zoomOutBtn).toBeTruthy() + + zoomOutBtn.triggerEventHandler('click', null) + expect(zoomViewSpy).toHaveBeenCalled() + const { args } = zoomViewSpy.calls.first() + expect(args[0]).toEqual(3) + /** + * zoom in < 1 + */ + expect(args[1]).toBeGreaterThan(1) + }) + + }) + }) }) -}) \ No newline at end of file +}) diff --git a/src/ui/nehubaContainer/nehubaContainer.component.ts b/src/ui/nehubaContainer/nehubaContainer.component.ts index 8d6726df10e10375c6a52e4a2b72418ca03bfc14..5e13e5789c093cff93384f47b7dcb34b4291e4f3 100644 --- a/src/ui/nehubaContainer/nehubaContainer.component.ts +++ b/src/ui/nehubaContainer/nehubaContainer.component.ts @@ -20,6 +20,7 @@ import { MOUSE_OVER_LANDMARK, NgViewerStateInterface } from "src/services/stateStore.service"; + import { getExportNehuba, isSame, getViewer } from "src/util/fn"; import { API_SERVICE_SET_VIEWER_HANDLE_TOKEN, IUserLandmark } from "src/atlasViewer/atlasViewer.apiService.service"; import { NehubaViewerUnit } from "./nehubaViewer/nehubaViewer.component"; @@ -33,11 +34,12 @@ import { viewerStateDblClickOnViewer, } from "src/services/state/viewerState.store.helper"; -import { getFourPanel, getHorizontalOneThree, getSinglePanel, getVerticalOneThree, calculateSliceZoomFactor, scanSliceViewRenderFn as scanFn } from "./util"; +import { getFourPanel, getHorizontalOneThree, getSinglePanel, getVerticalOneThree, calculateSliceZoomFactor, scanSliceViewRenderFn as scanFn, takeOnePipe } from "./util"; import { NehubaViewerContainerDirective } from "./nehubaViewerInterface/nehubaViewerInterface.directive"; import { ITunableProp } from "./mobileOverlay/mobileOverlay.component"; import {ConnectivityBrowserComponent} from "src/ui/connectivityBrowser/connectivityBrowser.component"; import { viewerStateMouseOverCustomLandmark } from "src/services/state/viewerState/actions"; +import { ngViewerSelectorNehubaReady, ngViewerSelectorOctantRemoval, ngViewerSelectorPanelMode, ngViewerSelectorPanelOrder } from "src/services/state/ngViewerState/selectors"; const { MESH_LOADING_STATUS } = IDS @@ -97,10 +99,11 @@ const sortByFreshness: (acc: any[], curr: any[]) => any[] = (acc, curr) => { const { ZOOM_IN, ZOOM_OUT, + TOGGLE_FRONTAL_OCTANT, TOGGLE_SIDE_PANEL, EXPAND, COLLAPSE, - ADDITIONAL_VOLUME_CONTROL + ADDITIONAL_VOLUME_CONTROL, } = ARIA_LABELS @Component({ @@ -149,6 +152,7 @@ export class NehubaContainer implements OnInit, OnChanges, OnDestroy { public CONST = CONST public ARIA_LABEL_ZOOM_IN = ZOOM_IN public ARIA_LABEL_ZOOM_OUT = ZOOM_OUT + public ARIA_LABEL_TOGGLE_FRONTAL_OCTANT = TOGGLE_FRONTAL_OCTANT public ARIA_LABEL_TOGGLE_SIDE_PANEL = TOGGLE_SIDE_PANEL public ARIA_LABEL_EXPAND = EXPAND public ARIA_LABEL_COLLAPSE = COLLAPSE @@ -299,33 +303,29 @@ export class NehubaContainer implements OnInit, OnChanges, OnDestroy { this.useMobileUI$ = this.pureConstantService.useTouchUI$ this.nehubaViewerPerspectiveOctantRemoval$ = this.store.pipe( - select('ngViewerState'), - select('octantRemoval') + select(ngViewerSelectorOctantRemoval), ) this.panelMode$ = this.store.pipe( - select('ngViewerState'), - select('panelMode'), + select(ngViewerSelectorPanelMode), distinctUntilChanged(), shareReplay(1), ) this.panelOrder$ = this.store.pipe( - select('ngViewerState'), - select('panelOrder'), + select(ngViewerSelectorPanelOrder), distinctUntilChanged(), shareReplay(1), ) this.redrawLayout$ = this.store.pipe( - select('ngViewerState'), - select('nehubaReady'), + select(ngViewerSelectorNehubaReady), distinctUntilChanged(), filter(v => !!v), - switchMapTo(combineLatest( + switchMapTo(combineLatest([ this.panelMode$, this.panelOrder$, - )), + ])), ) this.selectedLandmarks$ = this.store.pipe( @@ -661,7 +661,7 @@ export class NehubaContainer implements OnInit, OnChanges, OnDestroy { this.subscriptions.push( - combineLatest( + combineLatest([ this.selectedRegions$.pipe( distinctUntilChanged(), ), @@ -678,7 +678,7 @@ export class NehubaContainer implements OnInit, OnChanges, OnDestroy { select('overwrittenColorMap'), distinctUntilChanged() ) - ).pipe( + ]).pipe( delayWhen(() => timer()) ).subscribe(([regions, hideSegmentFlag, forceShowSegment, selectedParcellation, overwrittenColorMap]) => { if (!this.nehubaViewer) { return } @@ -1100,35 +1100,3 @@ export class NehubaContainer implements OnInit, OnChanges, OnDestroy { } } } - -export const takeOnePipe = () => { - - return pipe( - scan((acc: Event[], event: Event) => { - const target = (event as Event).target as HTMLElement - /** - * 0 | 1 - * 2 | 3 - * - * 4 ??? - */ - const panels = getViewer()['display']['panels'] - const panelEls = Array.from(panels).map(({ element }) => element) - - const identifySrcElement = (element: HTMLElement) => { - const idx = panelEls.indexOf(element) - return idx - } - - const key = identifySrcElement(target) - const _ = {} - _[key] = event - return Object.assign({}, acc, _) - }, []), - filter(v => { - const isdefined = (obj) => typeof obj !== 'undefined' && obj !== null - return (isdefined(v[0]) && isdefined(v[1]) && isdefined(v[2])) - }), - take(1), - ) -} diff --git a/src/ui/nehubaContainer/nehubaContainer.template.html b/src/ui/nehubaContainer/nehubaContainer.template.html index 6aee271815eb369c30ee9b753afaf8abc1355963..364093d9b98024013f18dabf06aec67bd51474f7 100644 --- a/src/ui/nehubaContainer/nehubaContainer.template.html +++ b/src/ui/nehubaContainer/nehubaContainer.template.html @@ -459,7 +459,10 @@ </div> <!-- maximise/minimise button --> - <ng-container *ngTemplateOutlet="panelCtrlTmpl; context: { panelIndex: 3, visible: (panelOrder$ | async | reorderPanelIndexPipe : ( hoveredPanelIndices$ | async )) === 3 }"> + <ng-container *ngTemplateOutlet="panelCtrlTmpl; context: { + panelIndex: panelOrder$ | async | getNthElement : 3 | parseAsNumber, + visible: (panelOrder$ | async | reorderPanelIndexPipe : ( hoveredPanelIndices$ | async )) === 3 + }"> </ng-container> <!-- mesh loading is still weird --> @@ -505,7 +508,10 @@ </nehuba-2dlandmark-unit> <!-- maximise/minimise button --> - <ng-container *ngTemplateOutlet="panelCtrlTmpl; context: { panelIndex: panelIndex, visible: (panelOrder$ | async | reorderPanelIndexPipe : ( hoveredPanelIndices$ | async )) === panelIndex }"> + <ng-container *ngTemplateOutlet="panelCtrlTmpl; context: { + panelIndex: panelOrder$ | async | getNthElement : panelIndex | parseAsNumber, + visible: (panelOrder$ | async | reorderPanelIndexPipe : ( hoveredPanelIndices$ | async )) === panelIndex + }"> </ng-container> <div *ngIf="(sliceViewLoadingMain$ | async)[panelIndex]" class="loadingIndicator"> @@ -523,7 +529,9 @@ let-visible="visible"> <div class="opacity-crossfade always-show-touchdevice pe-all overlay-btn-container" - [ngClass]="{ onHover: visible }"> + [ngClass]="{ onHover: visible }" + [attr.data-viewer-controller-visible]="visible" + [attr.data-viewer-controller-index]="panelIndex"> <!-- perspective specific control --> <ng-container *ngIf="panelIndex === 3"> @@ -559,6 +567,7 @@ <button (click)="setOctantRemoval(!state)" mat-icon-button + [attr.aria-label]="ARIA_LABEL_TOGGLE_FRONTAL_OCTANT" color="primary"> <!-- octant removal is true --> diff --git a/src/ui/nehubaContainer/nehubaViewerInterface/nehubaViewerInterface.directive.spec.ts b/src/ui/nehubaContainer/nehubaViewerInterface/nehubaViewerInterface.directive.spec.ts new file mode 100644 index 0000000000000000000000000000000000000000..91c8e5847a0b025b882057e7a2255fcad7bb5f7a --- /dev/null +++ b/src/ui/nehubaContainer/nehubaViewerInterface/nehubaViewerInterface.directive.spec.ts @@ -0,0 +1,148 @@ +import { CommonModule } from "@angular/common" +import { Component } from "@angular/core" +import { TestBed, async, ComponentFixture } from "@angular/core/testing" +import { By } from "@angular/platform-browser" +import { BrowserDynamicTestingModule } from "@angular/platform-browser-dynamic/testing" +import { MockStore, provideMockStore } from "@ngrx/store/testing" +import { ngViewerSelectorOctantRemoval } from "src/services/state/ngViewerState/selectors" +import { NehubaViewerUnit } from "../nehubaViewer/nehubaViewer.component" +import { NehubaViewerContainerDirective } from "./nehubaViewerInterface.directive" +import { viewerStateSelectorNavigation, viewerStateStandAloneVolumes } from "src/services/state/viewerState/selectors"; +import { Subject } from "rxjs" +import { ngViewerActionNehubaReady } from "src/services/state/ngViewerState/actions" + +describe('> nehubaViewerInterface.directive.ts', () => { + describe('> NehubaViewerContainerDirective', () => { + + @Component({ + template: '' + }) + class DummyCmp{} + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [ + + ], + declarations: [ + NehubaViewerContainerDirective, + DummyCmp, + NehubaViewerUnit, + ], + providers: [ + provideMockStore({ initialState: {} }) + ] + }).overrideModule(BrowserDynamicTestingModule,{ + set: { + entryComponents: [ + NehubaViewerUnit + ] + } + }).overrideComponent(DummyCmp, { + set: { + template: ` + <div iav-nehuba-viewer-container> + </div> + ` + } + }).compileComponents() + + })) + + beforeEach(() => { + const mockStore = TestBed.inject(MockStore) + mockStore.overrideSelector(ngViewerSelectorOctantRemoval, true) + mockStore.overrideSelector(viewerStateStandAloneVolumes, []) + mockStore.overrideSelector(viewerStateSelectorNavigation, null) + }) + + it('> can be inited', () => { + + const fixture = TestBed.createComponent(DummyCmp) + fixture.detectChanges() + const directive = fixture.debugElement.query( + By.directive(NehubaViewerContainerDirective) + ) + + expect(directive).toBeTruthy() + }) + + describe('> on createNehubaInstance', () => { + let fixture: ComponentFixture<DummyCmp> + let directiveInstance: NehubaViewerContainerDirective + let nehubaViewerInstanceSpy: jasmine.Spy + let elClearSpy: jasmine.Spy + let elCreateComponentSpy: jasmine.Spy + const spyNehubaViewerInstance = { + config: null, + lifecycle: null, + templateId: null, + errorEmitter: new Subject(), + debouncedViewerPositionChange: new Subject(), + layersChanged: new Subject(), + nehubaReady: new Subject(), + mouseoverSegmentEmitter: new Subject(), + mouseoverLandmarkEmitter: new Subject(), + mouseoverUserlandmarkEmitter: new Subject(), + elementRef: { + nativeElement: {} + } + } + const spyComRef = { + destroy: jasmine.createSpy('destroy') + } + + beforeEach(() => { + fixture = TestBed.createComponent(DummyCmp) + const directive = fixture.debugElement.query( + By.directive(NehubaViewerContainerDirective) + ) + + directiveInstance = directive.injector.get(NehubaViewerContainerDirective) + + nehubaViewerInstanceSpy = spyOnProperty(directiveInstance, 'nehubaViewerInstance').and.returnValue(spyNehubaViewerInstance) + elClearSpy = spyOn(directiveInstance['el'], 'clear') + // casting return value to any is not perfect, but since only 2 methods and 1 property is used, it's a quick way + // rather than allow component to be created + elCreateComponentSpy = spyOn(directiveInstance['el'], 'createComponent').and.returnValue(spyComRef as any) + }) + + describe('> on createNehubaInstance called', () => { + const template = {} + const lifecycle = {} + beforeEach(() => { + directiveInstance.createNehubaInstance(template, lifecycle) + }) + it('> method el.clear gets called before el.createComponent', () => { + expect(elClearSpy).toHaveBeenCalledBefore(elCreateComponentSpy) + }) + }) + + describe('> on clear called', () => { + it('> dispatches nehubaReady: false action', () => { + const mockStore = TestBed.inject(MockStore) + const mockStoreDispatchSpy = spyOn(mockStore, 'dispatch') + directiveInstance.clear() + expect( + mockStoreDispatchSpy + ).toHaveBeenCalledWith( + ngViewerActionNehubaReady({ + nehubaReady: false + }) + ) + }) + + it('> iavNehubaViewerContainerViewerLoading emits false', () => { + const emitSpy = spyOn(directiveInstance.iavNehubaViewerContainerViewerLoading, 'emit') + directiveInstance.clear() + expect(emitSpy).toHaveBeenCalledWith(false) + }) + + it('> elClear called', () => { + directiveInstance.clear() + expect(elClearSpy).toHaveBeenCalled() + }) + }) + }) + }) +}) diff --git a/src/ui/nehubaContainer/nehubaViewerInterface/nehubaViewerInterface.directive.ts b/src/ui/nehubaContainer/nehubaViewerInterface/nehubaViewerInterface.directive.ts index 74d8f40197f3b99d1d672888dea30c05ff03028f..49c2e9316963e3e19932f0da6e7af6c868ac91ca 100644 --- a/src/ui/nehubaContainer/nehubaViewerInterface/nehubaViewerInterface.directive.ts +++ b/src/ui/nehubaContainer/nehubaViewerInterface/nehubaViewerInterface.directive.ts @@ -5,14 +5,14 @@ import { IavRootStoreInterface } from "src/services/stateStore.service"; import { Subscription, Observable, fromEvent } from "rxjs"; import { distinctUntilChanged, filter, debounceTime, shareReplay, scan, map, throttleTime, switchMapTo } from "rxjs/operators"; import { StateInterface as ViewerConfigStateInterface } from "src/services/state/viewerConfig.store"; -import { getNavigationStateFromConfig } from "../util"; -import { NEHUBA_LAYER_CHANGED, CHANGE_NAVIGATION, VIEWERSTATE_ACTION_TYPES } from "src/services/state/viewerState.store"; +import { getNavigationStateFromConfig, takeOnePipe } from "../util"; +import { NEHUBA_LAYER_CHANGED, CHANGE_NAVIGATION } from "src/services/state/viewerState.store"; import { timedValues } from "src/util/generator"; import { MOUSE_OVER_SEGMENTS, MOUSE_OVER_LANDMARK } from "src/services/state/uiState.store"; -import { takeOnePipe } from "../nehubaContainer.component"; import { ngViewerActionNehubaReady } from "src/services/state/ngViewerState/actions"; import { viewerStateMouseOverCustomLandmarkInPerspectiveView } from "src/services/state/viewerState/actions"; -import { viewerStateStandAloneVolumes } from "src/services/state/viewerState/selectors"; +import { viewerStateStandAloneVolumes, viewerStateSelectorNavigation } from "src/services/state/viewerState/selectors"; +import { ngViewerSelectorOctantRemoval } from "src/services/state/ngViewerState/selectors"; const defaultNehubaConfig = { "configName": "", @@ -283,20 +283,14 @@ export class NehubaViewerContainerDirective implements OnInit, OnDestroy{ */ distinctUntilChanged(), ) - - const viewerState$ = this.store$.pipe( - select('viewerState'), - shareReplay(1) - ) - - this.navigationChanges$ = viewerState$.pipe( - select('navigation'), - filter(v => !!v) + + this.navigationChanges$ = this.store$.pipe( + select(viewerStateSelectorNavigation), + filter(v => !!v), ) this.nehubaViewerPerspectiveOctantRemoval$ = this.store$.pipe( - select('ngViewerState'), - select('octantRemoval') + select(ngViewerSelectorOctantRemoval), ) } @@ -482,6 +476,13 @@ export class NehubaViewerContainerDirective implements OnInit, OnDestroy{ while(this.nehubaViewerSubscriptions.length > 0) { this.nehubaViewerSubscriptions.pop().unsubscribe() } + + this.store$.dispatch( + ngViewerActionNehubaReady({ + nehubaReady: false, + }) + ) + this.iavNehubaViewerContainerViewerLoading.emit(false) if(this.cr) this.cr.destroy() this.el.clear() diff --git a/src/ui/nehubaContainer/touchSideClass.directive.ts b/src/ui/nehubaContainer/touchSideClass.directive.ts index f10e11d71845c5f3e61b62ec5140b80079975961..edaa9823ff6af39756e0f51e16c91cee40d91022 100644 --- a/src/ui/nehubaContainer/touchSideClass.directive.ts +++ b/src/ui/nehubaContainer/touchSideClass.directive.ts @@ -2,6 +2,7 @@ import { Directive, ElementRef, Input, OnDestroy, OnInit } from "@angular/core"; import { select, Store } from "@ngrx/store"; import { Observable, Subscription } from "rxjs"; import { distinctUntilChanged, tap } from "rxjs/operators"; +import { ngViewerSelectorPanelMode } from "src/services/state/ngViewerState/selectors"; import { IavRootStoreInterface } from "src/services/stateStore.service"; import { addTouchSideClasses, removeTouchSideClasses } from "./util"; @@ -25,8 +26,7 @@ export class TouchSideClass implements OnDestroy, OnInit { ) { this.panelMode$ = this.store$.pipe( - select('ngViewerState'), - select('panelMode'), + select(ngViewerSelectorPanelMode), distinctUntilChanged(), tap(mode => this.panelMode = mode), ) diff --git a/src/ui/nehubaContainer/util.ts b/src/ui/nehubaContainer/util.ts index 7efb0abcc211bfe381190467509e4d18ecbd2b0b..49d7136412e6d19d67bdf6774d7e6f81e9832988 100644 --- a/src/ui/nehubaContainer/util.ts +++ b/src/ui/nehubaContainer/util.ts @@ -1,4 +1,7 @@ +import { pipe } from 'rxjs' +import { filter, scan, take } from 'rxjs/operators' import { PANELS } from 'src/services/state/ngViewerState.store.helper' +import { getViewer } from 'src/util/fn' const flexContCmnCls = ['w-100', 'h-100', 'd-flex', 'justify-content-center', 'align-items-stretch'] @@ -250,3 +253,35 @@ export const scanSliceViewRenderFn: (acc: [boolean, boolean, boolean], curr: Cus } return returnAcc as [boolean, boolean, boolean] } + +export const takeOnePipe = () => { + + return pipe( + scan((acc: Event[], event: Event) => { + const target = (event as Event).target as HTMLElement + /** + * 0 | 1 + * 2 | 3 + * + * 4 ??? + */ + const panels = getViewer()['display']['panels'] + const panelEls = Array.from(panels).map(({ element }) => element) + + const identifySrcElement = (element: HTMLElement) => { + const idx = panelEls.indexOf(element) + return idx + } + + const key = identifySrcElement(target) + const _ = {} + _[key] = event + return Object.assign({}, acc, _) + }, []), + filter(v => { + const isdefined = (obj) => typeof obj !== 'undefined' && obj !== null + return (isdefined(v[0]) && isdefined(v[1]) && isdefined(v[2])) + }), + take(1), + ) +} diff --git a/src/util/pipes/getNthElement.pipe.ts b/src/util/pipes/getNthElement.pipe.ts new file mode 100644 index 0000000000000000000000000000000000000000..d4b27720b903731fa8c5cb0526380030854b7d4d --- /dev/null +++ b/src/util/pipes/getNthElement.pipe.ts @@ -0,0 +1,11 @@ +import { Pipe, PipeTransform } from "@angular/core"; + +@Pipe({ + name: 'getNthElement' +}) +export class GetNthElementPipe<T> implements PipeTransform{ + public transform(array: T[], idx: number): T{ + if (!array[idx]) throw new Error(`[GetNthElementPipe] accessor error`) + return array[idx] + } +} diff --git a/src/util/pipes/parseAsNumber.pipe.ts b/src/util/pipes/parseAsNumber.pipe.ts new file mode 100644 index 0000000000000000000000000000000000000000..40688b6b9847f3fafce2d8c1dddfb64e1355550a --- /dev/null +++ b/src/util/pipes/parseAsNumber.pipe.ts @@ -0,0 +1,12 @@ +import { Pipe, PipeTransform } from "@angular/core"; + +@Pipe({ + name: 'parseAsNumber' +}) + +export class ParseAsNumberPipe implements PipeTransform{ + public transform(input: string | string[]): number | number[]{ + if (input instanceof Array) return input.map(v => Number(v)) + return Number(input) + } +} diff --git a/src/util/util.module.ts b/src/util/util.module.ts index 4c8bf1ce5bc6b597e1bc9229c02f48df74c6d949..f29aa368ea5931808431ac7f5ca87f3761655c04 100644 --- a/src/util/util.module.ts +++ b/src/util/util.module.ts @@ -16,6 +16,8 @@ import { LayoutModule } from "@angular/cdk/layout"; import { MapToPropertyPipe } from "./pipes/mapToProperty.pipe"; import {ClickOutsideDirective} from "src/util/directives/clickOutside.directive"; import { CounterDirective } from "./directives/counter.directive"; +import { GetNthElementPipe } from "./pipes/getNthElement.pipe"; +import { ParseAsNumberPipe } from "./pipes/parseAsNumber.pipe"; @NgModule({ imports:[ @@ -36,7 +38,9 @@ import { CounterDirective } from "./directives/counter.directive"; MediaQueryDirective, MapToPropertyPipe, ClickOutsideDirective, - CounterDirective + CounterDirective, + GetNthElementPipe, + ParseAsNumberPipe, ], exports: [ FilterNullPipe, @@ -53,7 +57,9 @@ import { CounterDirective } from "./directives/counter.directive"; MediaQueryDirective, MapToPropertyPipe, ClickOutsideDirective, - CounterDirective + CounterDirective, + GetNthElementPipe, + ParseAsNumberPipe, ] })