Skip to content
Snippets Groups Projects
Commit 29c31f41 authored by Xiao Gui's avatar Xiao Gui
Browse files

bugfix: status panel causes nav errors (#707)

parent 4e43734d
No related branches found
No related tags found
No related merge requests found
...@@ -25,6 +25,7 @@ import { ...@@ -25,6 +25,7 @@ import {
viewerStateNewViewer viewerStateNewViewer
} from './viewerState.store.helper'; } from './viewerState.store.helper';
import { cvtNehubaConfigToNavigationObj } from 'src/ui/viewerStateController/viewerState.useEffect'; import { cvtNehubaConfigToNavigationObj } from 'src/ui/viewerStateController/viewerState.useEffect';
import { viewerStateChangeNavigation } from './viewerState/actions';
export interface StateInterface { export interface StateInterface {
fetchedTemplates: any[] fetchedTemplates: any[]
...@@ -162,6 +163,7 @@ export const getStateStore = ({ state = defaultState } = {}) => (prevState: Part ...@@ -162,6 +163,7 @@ export const getStateStore = ({ state = defaultState } = {}) => (prevState: Part
fetchedTemplates: prevState.fetchedTemplates.concat(action.fetchedTemplate), fetchedTemplates: prevState.fetchedTemplates.concat(action.fetchedTemplate),
} }
} }
case viewerStateChangeNavigation.type:
case CHANGE_NAVIGATION : { case CHANGE_NAVIGATION : {
return { return {
...prevState, ...prevState,
...@@ -271,7 +273,7 @@ export const UNLOAD_DEDICATED_LAYER = 'UNLOAD_DEDICATED_LAYER' ...@@ -271,7 +273,7 @@ export const UNLOAD_DEDICATED_LAYER = 'UNLOAD_DEDICATED_LAYER'
export const NEWVIEWER = viewerStateNewViewer.type export const NEWVIEWER = viewerStateNewViewer.type
export const FETCHED_TEMPLATE = 'FETCHED_TEMPLATE' export const FETCHED_TEMPLATE = 'FETCHED_TEMPLATE'
export const CHANGE_NAVIGATION = 'CHANGE_NAVIGATION' export const CHANGE_NAVIGATION = viewerStateChangeNavigation.type
export const SELECT_PARCELLATION = viewerStateSelectParcellation.type export const SELECT_PARCELLATION = viewerStateSelectParcellation.type
......
...@@ -105,3 +105,7 @@ export const viewerStateMouseOverCustomLandmarkInPerspectiveView = createAction( ...@@ -105,3 +105,7 @@ export const viewerStateMouseOverCustomLandmarkInPerspectiveView = createAction(
props<{ payload: { label: string } }>() props<{ payload: { label: string } }>()
) )
export const viewerStateChangeNavigation = createAction(
`[viewerState] changeNavigation`,
props<{ navigation: any }>()
)
import { createSelector } from "@ngrx/store" import { createSelector } from "@ngrx/store"
import { create } from "domain"
import { viewerStateHelperStoreName } from "../viewerState.store.helper" import { viewerStateHelperStoreName } from "../viewerState.store.helper"
export const viewerStateSelectedRegionsSelector = createSelector( export const viewerStateSelectedRegionsSelector = createSelector(
...@@ -32,6 +33,18 @@ export const viewerStateSelectedTemplateSelector = createSelector( ...@@ -32,6 +33,18 @@ export const viewerStateSelectedTemplateSelector = createSelector(
viewerState => viewerState['templateSelected'] viewerState => viewerState['templateSelected']
) )
/**
* viewerStateSelectedTemplateSelector may have it navigation mutated to allow for initiliasation of viewer at the correct navigation
* in some circumstances, it may be required to get the original navigation object
*/
export const viewerStateSelectedTemplatePureSelector = createSelector(
viewerStateFetchedTemplatesSelector,
viewerStateSelectedTemplateSelector,
(fetchedTemplates, selectedTemplate) => {
return fetchedTemplates.find(t => t['@id'] === selectedTemplate('@id'))
}
)
export const viewerStateSelectedParcellationSelector = createSelector( export const viewerStateSelectedParcellationSelector = createSelector(
state => state['viewerState'], state => state['viewerState'],
viewerState => viewerState['parcellationSelected'] viewerState => viewerState['parcellationSelected']
......
import { async, TestBed } from "@angular/core/testing" import { async, ComponentFixture, TestBed } from "@angular/core/testing"
import { CommonModule } from "@angular/common" import { CommonModule } from "@angular/common"
import { AngularMaterialModule } from "src/ui/sharedModules/angularMaterial.module" import { AngularMaterialModule } from "src/ui/sharedModules/angularMaterial.module"
import { StatusCardComponent } from "./statusCard.component" import { StatusCardComponent } from "./statusCard.component"
import { Directive, Component } from "@angular/core" import { Directive, Component } from "@angular/core"
import { of } from "rxjs" import { Observable, of } from "rxjs"
import { ShareModule } from "src/share" import { ShareModule } from "src/share"
import { StateModule } from "src/state" import { StateModule } from "src/state"
import { MockStore, provideMockStore } from "@ngrx/store/testing" import { MockStore, provideMockStore } from "@ngrx/store/testing"
...@@ -13,6 +13,9 @@ import { NoopAnimationsModule } from "@angular/platform-browser/animations" ...@@ -13,6 +13,9 @@ import { NoopAnimationsModule } from "@angular/platform-browser/animations"
import { FormsModule, ReactiveFormsModule } from "@angular/forms" import { FormsModule, ReactiveFormsModule } from "@angular/forms"
import { UtilModule } from "src/util" import { UtilModule } from "src/util"
import { viewerConfigSelectorUseMobileUi } from "src/services/state/viewerConfig.store.helper" import { viewerConfigSelectorUseMobileUi } from "src/services/state/viewerConfig.store.helper"
import { viewerStateNavigationStateSelector, viewerStateSelectedTemplatePureSelector } from "src/services/state/viewerState/selectors"
import * as util from '../util'
import { viewerStateChangeNavigation } from "src/services/state/viewerState/actions"
@Directive({ @Directive({
selector: '[iav-auth-authState]', selector: '[iav-auth-authState]',
...@@ -73,55 +76,126 @@ describe('> statusCard.component.ts', () => { ...@@ -73,55 +76,126 @@ describe('> statusCard.component.ts', () => {
expect(fixture.debugElement.nativeElement).toBeTruthy() expect(fixture.debugElement.nativeElement).toBeTruthy()
}) })
it('> toggle can be found if showFull is set to true', () => { describe('> in full mode, UIs are visible', () => {
let fixture: ComponentFixture<StatusCardComponent>
const fixture = TestBed.createComponent(StatusCardComponent)
fixture.detectChanges()
fixture.componentInstance.showFull = true
fixture.detectChanges()
const slider = fixture.debugElement.query( By.directive(MatSlideToggle) )
expect(slider).toBeTruthy()
})
it('> toggling voxel/real toggle also toggles statusPanelRealSpace flag', () => { beforeEach(() => {
const fixture = TestBed.createComponent(StatusCardComponent) fixture = TestBed.createComponent(StatusCardComponent)
fixture.detectChanges() fixture.detectChanges()
fixture.componentInstance.showFull = true fixture.componentInstance.showFull = true
fixture.detectChanges() fixture.detectChanges()
})
it('> toggle can be found', () => {
const prevFlag = fixture.componentInstance.statusPanelRealSpace const slider = fixture.debugElement.query( By.directive(MatSlideToggle) )
const sliderEl = fixture.debugElement.query( By.directive(MatSlideToggle) ) expect(slider).toBeTruthy()
const slider = sliderEl.injector.get(MatSlideToggle) })
slider.toggle()
fixture.detectChanges() it('> toggling voxel/real toggle also toggles statusPanelRealSpace flag', () => {
expect(fixture.componentInstance.statusPanelRealSpace).toEqual(!prevFlag)
const prevFlag = fixture.componentInstance.statusPanelRealSpace
const sliderEl = fixture.debugElement.query( By.directive(MatSlideToggle) )
const slider = sliderEl.injector.get(MatSlideToggle)
slider.toggle()
fixture.detectChanges()
expect(fixture.componentInstance.statusPanelRealSpace).toEqual(!prevFlag)
})
describe('> textNavigationTo', () => {
it('> takes into account of statusPanelRealSpace panel', () => {
const setNavigationStateSpy = jasmine.createSpy('setNavigationState')
fixture.componentInstance.nehubaViewer = {
setNavigationState: setNavigationStateSpy
} as any
fixture.componentInstance.statusPanelRealSpace = true
fixture.componentInstance.textNavigateTo('1, 0, 0')
expect(setNavigationStateSpy).toHaveBeenCalledWith({
position: [1e6, 0, 0],
positionReal: true
})
fixture.componentInstance.statusPanelRealSpace = false
fixture.componentInstance.textNavigateTo('1, 0, 0')
expect(setNavigationStateSpy).toHaveBeenCalledWith({
position: [1, 0, 0],
positionReal: false
})
})
})
}) })
describe('> textNavigationTo', () => { describe('> resetNavigation', () => {
it('> takes into account of statusPanelRealSpace panel', () => { let fixture: ComponentFixture<StatusCardComponent>
const fixture = TestBed.createComponent(StatusCardComponent) const mockCurrNavigation = {
orientation: [1,0,0,0],
position: [100,200,300],
perspectiveZoom: 1e9,
zoom: 1e9,
perspectiveOrientation: [1,0,0,0]
}
const mockNavState = {
orientation: [0,0,0,1],
position: [10,20,30],
perspectiveZoom: 1e6,
zoom: 1e6,
perspectiveOrientation: [0,0,0,1]
}
const mockTemplate = { foo:'bar', nehubaConfig: { foo2: 'bar2' } }
let getNavigationStateFromConfigSpy: jasmine.Spy = jasmine.createSpy('getNavigationStateFromConfig').and.returnValue(mockNavState)
beforeEach(() => {
const mockStore = TestBed.inject(MockStore)
mockStore.overrideSelector(viewerStateSelectedTemplatePureSelector, mockTemplate)
mockStore.overrideSelector(viewerStateNavigationStateSelector, mockCurrNavigation)
spyOnProperty(util, 'getNavigationStateFromConfig').and.returnValue(getNavigationStateFromConfigSpy)
fixture = TestBed.createComponent(StatusCardComponent)
fixture.detectChanges() fixture.detectChanges()
const setNavigationStateSpy = jasmine.createSpy('setNavigationState') fixture.componentInstance.showFull = true
fixture.componentInstance.nehubaViewer = { fixture.detectChanges()
setNavigationState: setNavigationStateSpy })
} as any
for (const method of ['rotation', 'position', 'zoom' ]) {
fixture.componentInstance.statusPanelRealSpace = true describe(`> method: ${method}`, () => {
fixture.componentInstance.textNavigateTo('1, 0, 0')
expect(setNavigationStateSpy).toHaveBeenCalledWith({ it(`> resetNavigation call calls getNavigationStateFromConfig`, () => {
position: [1e6, 0, 0], fixture.componentInstance.resetNavigation({ [method]: true, })
positionReal: true fixture.detectChanges()
}) expect(getNavigationStateFromConfigSpy).toHaveBeenCalled()
})
fixture.componentInstance.statusPanelRealSpace = false it('> resetNavigation dispatches correct action', () => {
fixture.componentInstance.textNavigateTo('1, 0, 0')
expect(setNavigationStateSpy).toHaveBeenCalledWith({ const mockStore = TestBed.inject(MockStore)
position: [1, 0, 0], const idspatchSpy = spyOn(mockStore, 'dispatch')
positionReal: false fixture.componentInstance.resetNavigation({ [method]: true, })
fixture.detectChanges()
const overrideObj = {}
if (method === 'rotation') overrideObj['orientation'] = mockNavState['orientation']
if (method === 'position') overrideObj['position'] = mockNavState['position']
if (method === 'zoom') overrideObj['zoom'] = mockNavState['zoom']
expect(idspatchSpy).toHaveBeenCalledWith(
viewerStateChangeNavigation({
navigation: {
...mockCurrNavigation,
...overrideObj,
positionReal: false,
animation: {},
}
})
)
})
}) })
}) }
}) })
}) })
}) })
import { Component, Input, OnInit, OnChanges, TemplateRef, HostBinding } from "@angular/core"; import { Component, Input, OnInit, OnChanges, TemplateRef, HostBinding } from "@angular/core";
import { select, Store } from "@ngrx/store"; import { select, Store } from "@ngrx/store";
import { LoggingService } from "src/logging"; import { LoggingService } from "src/logging";
import { CHANGE_NAVIGATION, IavRootStoreInterface, ViewerStateInterface } from "src/services/stateStore.service"; import { IavRootStoreInterface } from "src/services/stateStore.service";
import { NehubaViewerUnit } from "../nehubaViewer/nehubaViewer.component"; import { NehubaViewerUnit } from "../nehubaViewer/nehubaViewer.component";
import { Observable, Subscription, of, combineLatest, BehaviorSubject } from "rxjs"; import { Observable, Subscription, of, combineLatest } from "rxjs";
import { distinctUntilChanged, shareReplay, map, filter, startWith } from "rxjs/operators"; import { map, filter, startWith } from "rxjs/operators";
import { MatBottomSheet } from "@angular/material/bottom-sheet"; import { MatBottomSheet } from "@angular/material/bottom-sheet";
import { MatDialog } from "@angular/material/dialog"; import { MatDialog } from "@angular/material/dialog";
import { ARIA_LABELS } from 'common/constants' import { ARIA_LABELS } from 'common/constants'
import { PureContantService } from "src/util"; import { PureContantService } from "src/util";
import { FormControl } from "@angular/forms"; import { FormControl } from "@angular/forms";
import { viewerStateNavigationStateSelector, viewerStateSelectedTemplatePureSelector } from "src/services/state/viewerState/selectors";
import { getNavigationStateFromConfig } from "../util";
import { viewerStateChangeNavigation } from "src/services/state/viewerState/actions";
@Component({ @Component({
selector : 'ui-status-card', selector : 'ui-status-card',
...@@ -25,8 +28,8 @@ export class StatusCardComponent implements OnInit, OnChanges{ ...@@ -25,8 +28,8 @@ export class StatusCardComponent implements OnInit, OnChanges{
public arialabel = ARIA_LABELS.STATUS_PANEL public arialabel = ARIA_LABELS.STATUS_PANEL
public showFull = false public showFull = false
private selectedTemplateRoot$: Observable<any> private selectedTemplatePure: any
private selectedTemplateRoot: any private currentNavigation: any
private subscriptions: Subscription[] = [] private subscriptions: Subscription[] = []
public navVal$: Observable<string> public navVal$: Observable<string>
...@@ -41,36 +44,32 @@ export class StatusCardComponent implements OnInit, OnChanges{ ...@@ -41,36 +44,32 @@ export class StatusCardComponent implements OnInit, OnChanges{
public SHOW_FULL_STATUS_PANEL_ARIA_LABEL = ARIA_LABELS.SHOW_FULL_STATUS_PANEL public SHOW_FULL_STATUS_PANEL_ARIA_LABEL = ARIA_LABELS.SHOW_FULL_STATUS_PANEL
public HIDE_FULL_STATUS_PANEL_ARIA_LABEL = ARIA_LABELS.HIDE_FULL_STATUS_PANEL public HIDE_FULL_STATUS_PANEL_ARIA_LABEL = ARIA_LABELS.HIDE_FULL_STATUS_PANEL
constructor( constructor(
private store: Store<ViewerStateInterface>,
private log: LoggingService,
private store$: Store<IavRootStoreInterface>, private store$: Store<IavRootStoreInterface>,
private log: LoggingService,
private bottomSheet: MatBottomSheet, private bottomSheet: MatBottomSheet,
private dialog: MatDialog, private dialog: MatDialog,
private pureConstantService: PureContantService private pureConstantService: PureContantService
) { ) {
const viewerState$ = this.store$.pipe(
select('viewerState'),
shareReplay(1),
)
this.selectedTemplateRoot$ = viewerState$.pipe(
select('fetchedTemplates'),
distinctUntilChanged(),
)
this.useTouchInterface$ = this.pureConstantService.useTouchUI$ this.useTouchInterface$ = this.pureConstantService.useTouchUI$
} }
ngOnInit(): void { ngOnInit(): void {
this.subscriptions.push( this.subscriptions.push(
this.selectedTemplateRoot$.subscribe(template => { this.statusPanelFormCtrl.valueChanges.subscribe(val => {
this.selectedTemplateRoot = template.find(t => t.name === this.selectedTemplateName) this.statusPanelRealSpace = val
}) })
) )
this.subscriptions.push( this.subscriptions.push(
this.statusPanelFormCtrl.valueChanges.subscribe(val => { this.store$.pipe(
this.statusPanelRealSpace = val select(viewerStateSelectedTemplatePureSelector)
}) ).subscribe(n => this.selectedTemplatePure = n)
)
this.subscriptions.push(
this.store$.pipe(
select(viewerStateNavigationStateSelector)
).subscribe(nav => this.currentNavigation = nav)
) )
} }
...@@ -80,7 +79,7 @@ export class StatusCardComponent implements OnInit, OnChanges{ ...@@ -80,7 +79,7 @@ export class StatusCardComponent implements OnInit, OnChanges{
this.mouseVal$ = of(`neubaViewer is undefined`) this.mouseVal$ = of(`neubaViewer is undefined`)
return return
} }
this.navVal$ = combineLatest( this.navVal$ = combineLatest([
this.statusPanelRealSpace$, this.statusPanelRealSpace$,
this.nehubaViewer.viewerPosInReal$.pipe( this.nehubaViewer.viewerPosInReal$.pipe(
filter(v => !!v) filter(v => !!v)
...@@ -88,14 +87,14 @@ export class StatusCardComponent implements OnInit, OnChanges{ ...@@ -88,14 +87,14 @@ export class StatusCardComponent implements OnInit, OnChanges{
this.nehubaViewer.viewerPosInVoxel$.pipe( this.nehubaViewer.viewerPosInVoxel$.pipe(
filter(v => !!v) filter(v => !!v)
) )
).pipe( ]).pipe(
map(([realFlag, real, voxel]) => realFlag map(([realFlag, real, voxel]) => realFlag
? real.map(v => `${ (v / 1e6).toFixed(3) }mm`).join(', ') ? real.map(v => `${ (v / 1e6).toFixed(3) }mm`).join(', ')
: voxel.map(v => v.toFixed(3)).join(', ') ), : voxel.map(v => v.toFixed(3)).join(', ') ),
startWith(`nehubaViewer initialising`) startWith(`nehubaViewer initialising`)
) )
this.mouseVal$ = combineLatest( this.mouseVal$ = combineLatest([
this.statusPanelRealSpace$, this.statusPanelRealSpace$,
this.nehubaViewer.mousePosInReal$.pipe( this.nehubaViewer.mousePosInReal$.pipe(
filter(v => !!v) filter(v => !!v)
...@@ -103,7 +102,7 @@ export class StatusCardComponent implements OnInit, OnChanges{ ...@@ -103,7 +102,7 @@ export class StatusCardComponent implements OnInit, OnChanges{
this.nehubaViewer.mousePosInVoxel$.pipe( this.nehubaViewer.mousePosInVoxel$.pipe(
filter(v => !!v) filter(v => !!v)
) )
).pipe( ]).pipe(
map(([realFlag, real, voxel]) => realFlag map(([realFlag, real, voxel]) => realFlag
? real.map(v => `${ (v/1e6).toFixed(3) }mm`).join(', ') ? real.map(v => `${ (v/1e6).toFixed(3) }mm`).join(', ')
: voxel.map(v => v.toFixed(3)).join(', ')), : voxel.map(v => v.toFixed(3)).join(', ')),
...@@ -111,7 +110,7 @@ export class StatusCardComponent implements OnInit, OnChanges{ ...@@ -111,7 +110,7 @@ export class StatusCardComponent implements OnInit, OnChanges{
) )
} }
statusPanelFormCtrl = new FormControl(true, []) public statusPanelFormCtrl = new FormControl(true, [])
public statusPanelRealSpace = true public statusPanelRealSpace = true
public statusPanelRealSpace$ = this.statusPanelFormCtrl.valueChanges.pipe( public statusPanelRealSpace$ = this.statusPanelFormCtrl.valueChanges.pipe(
startWith(true) startWith(true)
...@@ -143,44 +142,26 @@ export class StatusCardComponent implements OnInit, OnChanges{ ...@@ -143,44 +142,26 @@ export class StatusCardComponent implements OnInit, OnChanges{
* the info re: nehubaViewer can stay there, too * the info re: nehubaViewer can stay there, too
*/ */
public resetNavigation({rotation: rotationFlag = false, position: positionFlag = false, zoom : zoomFlag = false}: {rotation?: boolean, position?: boolean, zoom?: boolean}) { public resetNavigation({rotation: rotationFlag = false, position: positionFlag = false, zoom : zoomFlag = false}: {rotation?: boolean, position?: boolean, zoom?: boolean}) {
const initialNgState = this.selectedTemplateRoot.nehubaConfig.dataset.initialNgState // d sa dsa const {
orientation,
const perspectiveZoom = initialNgState ? initialNgState.perspectiveZoom : undefined perspectiveOrientation,
const perspectiveOrientation = initialNgState ? initialNgState.perspectiveOrientation : undefined perspectiveZoom,
const zoom = (zoomFlag position,
&& initialNgState zoom
&& initialNgState.navigation } = getNavigationStateFromConfig(this.selectedTemplatePure.nehubaConfig)
&& initialNgState.navigation.zoomFactor)
|| undefined this.store$.dispatch(
viewerStateChangeNavigation({
const position = (positionFlag navigation: {
&& initialNgState ...this.currentNavigation,
&& initialNgState.navigation ...(rotationFlag ? { orientation: orientation } : {}),
&& initialNgState.navigation.pose ...(positionFlag ? { position: position } : {}),
&& initialNgState.navigation.pose.position.voxelCoordinates ...(zoomFlag ? { zoom: zoom } : {}),
&& initialNgState.navigation.pose.position.voxelCoordinates)
|| undefined
const orientation = rotationFlag
? [0, 0, 0, 1]
: undefined
this.store.dispatch({
type : CHANGE_NAVIGATION,
navigation : {
...{
perspectiveZoom,
perspectiveOrientation,
zoom,
position,
orientation,
},
...{
positionReal : false, positionReal : false,
animation : {}, animation : {},
}, }
}, })
}) )
} }
openDialog(tmpl: TemplateRef<any>, options) { openDialog(tmpl: TemplateRef<any>, options) {
......
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment