diff --git a/common/constants.js b/common/constants.js index 0b92fb901782e568b9c8f2db52f9f7c9a79156b3..d3fe2b2b308520de3e5e9142d58b681c9dbc2a5a 100644 --- a/common/constants.js +++ b/common/constants.js @@ -163,5 +163,6 @@ If you do not accept the Terms & Conditions you are not permitted to access or u } exports.QUICKTOUR_DESC_MD = { + SLICE_VIEW: `The planar views allow you to zoom \`[mouse wheel]\`, pan the view \`[drag]\`, and select oblique sections \`<shift>\` + \`[drag]\`. You can \`[click]\` any brain regions to select them.` } })(typeof exports === 'undefined' ? module.exports : exports) diff --git a/deploy/csp/index.js b/deploy/csp/index.js index e976dd482d2845490dd24bebef212ea1ab0aeb17..e48f7f155e90e027e0274590f854d7281ca83bad 100644 --- a/deploy/csp/index.js +++ b/deploy/csp/index.js @@ -115,7 +115,7 @@ module.exports = { 'https://unpkg.com/d3@6.2.0/', // required for preview component 'https://unpkg.com/mathjax@3.1.2/', // math jax 'https://unpkg.com/three-surfer@0.0.13/dist/bundle.js', // for threeSurfer (freesurfer support in browser) - 'https://unpkg.com/ng-layer-tune@0.0.12/dist/ng-layer-tune/', // needed for ng layer control + 'https://unpkg.com/ng-layer-tune@0.0.13/dist/ng-layer-tune/', // needed for ng layer control 'https://unpkg.com/hbp-connectivity-component@0.6.6/', // needed for connectivity component (req, res) => res.locals.nonce ? `'nonce-${res.locals.nonce}'` : null, ...SCRIPT_SRC, diff --git a/docs/releases/v2.11.2.md b/docs/releases/v2.11.2.md new file mode 100644 index 0000000000000000000000000000000000000000..cd4023f5a6b472d7942fb5badd23c95b62430530 --- /dev/null +++ b/docs/releases/v2.11.2.md @@ -0,0 +1,15 @@ +# v2.11.2 + +## Features + +- Allow external layer control to persist state (#1338) + + +## Bugfixes + +- Fixed neuron display +- Fixed typos in plugin API messages + +## Behind the scenes + +- Pass messages to plugin API when not handled \ No newline at end of file diff --git a/mkdocs.yml b/mkdocs.yml index df2d3dcc9969d727dc219a4a2450f0da70533330..fc88fe80be9e51f1591d52834ece4a8698b79ef5 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -33,6 +33,7 @@ nav: - Fetching datasets: 'advanced/datasets.md' - Display non-atlas volumes: 'advanced/otherVolumes.md' - Release notes: + - v2.11.2: 'releases/v2.11.2.md' - v2.11.1: 'releases/v2.11.1.md' - v2.11.0: 'releases/v2.11.0.md' - v2.10.3: 'releases/v2.10.3.md' diff --git a/package.json b/package.json index 9719e8ac3d423f69b3e79bc3c266675522175327..4ea49304574b3072c6f16f5690b98a624b3558ba 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "siibra-explorer", - "version": "2.11.1", + "version": "2.11.2", "description": "siibra-explorer - explore brain atlases. Based on humanbrainproject/nehuba & google/neuroglancer. Built with angular", "scripts": { "lint": "eslint src --ext .ts", diff --git a/src/api/service.ts b/src/api/service.ts index 6adceb099682923f5306647b857346dca23b1aa3..44520f13f9a2958437463fb45793a925ee5b5aec 100644 --- a/src/api/service.ts +++ b/src/api/service.ts @@ -172,7 +172,7 @@ const broadCastDefault: BroadCastingApiEvents = { export class ApiService implements BoothResponder<ApiBoothEvents>{ - public broadcastCh = createBroadcastingJsonRpcChannel<`${NAMESPACE_TYPE}.on`, BroadCastingApiEvents>(`${namespace}.on`, broadCastDefault) + public broadcastCh = createBroadcastingJsonRpcChannel<`${NAMESPACE_TYPE}.on.`, BroadCastingApiEvents>(`${namespace}.on.`, broadCastDefault) public booth = new Booth<ApiBoothEvents>(this) private requestUserQueue: RequestUser<keyof RequestUserTypes>[] = [] @@ -263,27 +263,27 @@ export class ApiService implements BoothResponder<ApiBoothEvents>{ this.store.pipe( select(atlasSelection.selectors.selectedAtlas) ).subscribe(atlas => { - this.broadcastCh.emit('atlasSelected', translateV3Entities.retrieveAtlas(atlas)) + this.broadcastCh.emit('atlasSelected', atlas && translateV3Entities.retrieveAtlas(atlas)) }) this.store.pipe( select(atlasSelection.selectors.selectedParcellation) ).subscribe(parcellation => { - this.broadcastCh.emit('parcellationSelected', translateV3Entities.retrieveParcellation(parcellation)) + this.broadcastCh.emit('parcellationSelected', parcellation && translateV3Entities.retrieveParcellation(parcellation)) }) this.store.pipe( select(atlasSelection.selectors.selectedTemplate) ).subscribe(template => { - this.broadcastCh.emit('templateSelected', translateV3Entities.retrieveTemplate(template)) + this.broadcastCh.emit('templateSelected', template && translateV3Entities.retrieveTemplate(template)) }) this.store.pipe( select(atlasSelection.selectors.selectedRegions) ).subscribe(regions => { - this.broadcastCh.emit('regionsSelected', regions.map(reg => translateV3Entities.retrieveRegion(reg))) + this.broadcastCh.emit('regionsSelected', regions && regions.map(reg => translateV3Entities.retrieveRegion(reg))) }) this.store.pipe( select(atlasSelection.selectors.selectedParcAllRegions) ).subscribe(regions => { - this.broadcastCh.emit('allRegions', regions.map(reg => translateV3Entities.retrieveRegion(reg))) + this.broadcastCh.emit('allRegions', regions && regions.map(reg => translateV3Entities.retrieveRegion(reg))) }) this.store.pipe( select(atlasSelection.selectors.navigation) diff --git a/src/features/entry/entry.component.ts b/src/features/entry/entry.component.ts index 722fd8e99735fb88a04ab0435c6a0bccaa760306..439deee00bfe54122ca8bd0df9ce38359019f362 100644 --- a/src/features/entry/entry.component.ts +++ b/src/features/entry/entry.component.ts @@ -128,7 +128,6 @@ export class EntryComponent extends FeatureBase implements AfterViewInit, OnDest await ds.pull() } catch (e) { if (e instanceof DsExhausted) { - console.log('exhausted') break } if (e instanceof IsAlreadyPulling ) { diff --git a/src/index.html b/src/index.html index c1cef937495c9ed9218e6ec90326307b07a8816e..1ed2d8187899e51442bc39c8ab9305ac15c7ea0f 100644 --- a/src/index.html +++ b/src/index.html @@ -14,7 +14,7 @@ <script src="extra_js.js"></script> <script src="https://unpkg.com/kg-dataset-previewer@1.2.0/dist/kg-dataset-previewer/kg-dataset-previewer.js" defer></script> <script src="https://unpkg.com/three-surfer@0.0.13/dist/bundle.js" defer></script> - <script type="module" src="https://unpkg.com/ng-layer-tune@0.0.12/dist/ng-layer-tune/ng-layer-tune.esm.js"></script> + <script type="module" src="https://unpkg.com/ng-layer-tune@0.0.13/dist/ng-layer-tune/ng-layer-tune.esm.js"></script> <script type="module" src="https://unpkg.com/hbp-connectivity-component@0.6.6/dist/connectivity-component/connectivity-component.js" ></script> <script defer src="https://unpkg.com/mathjax@3.1.2/es5/tex-svg.js"></script> <script defer src="https://unpkg.com/d3@6.2.0/dist/d3.min.js"></script> diff --git a/src/messaging/nmvSwc/index.ts b/src/messaging/nmvSwc/index.ts index d7bd4d6a767d013e2e357cb98d79202a7ecfed08..8d5d824def87226d1f08265340d7120a22b626dc 100644 --- a/src/messaging/nmvSwc/index.ts +++ b/src/messaging/nmvSwc/index.ts @@ -162,6 +162,9 @@ export const processJsonLd = (json: { [key: string]: any }): Observable<IMessagi * swc seem to scale with voxelSize... strangely enough * voxelSize nm / voxel -> goal is 1 voxel/um * 1e3 / voxelSize + * + * update: nope... it seems ... at least the SWC sent so far + * */ const scaleUmToVoxelFixed = [ voxelSize[0], @@ -171,12 +174,13 @@ export const processJsonLd = (json: { [key: string]: any }): Observable<IMessagi // NG translation works on nm scale const scaleUmToNm = 1e3 const modA = mat3.fromValues( - scaleUmToVoxelFixed[0], 0, 0, - 0, scaleUmToVoxelFixed[1], 0, - 0, 0, scaleUmToVoxelFixed[2] + scaleUmToNm, 0, 0, + 0, scaleUmToNm, 0, + 0, 0, scaleUmToNm ) mat3.mul(modA, modA, [...A[0], ...A[1], ...A[2]]) - const modb = vec3.scale(vec3.create(), b, scaleUmToNm) + const modb = vec3.mul(vec3.create(), b, [ scaleUmToNm, scaleUmToNm, scaleUmToNm]) + vec3.scale(vec3.create(), b, scaleUmToNm) const transform = [ [...modA.slice(0, 3), modb[0]] as TVec4, [...modA.slice(3, 6), modb[1]] as TVec4, diff --git a/src/messaging/service.ts b/src/messaging/service.ts index 365a2d1880bd4513bb088b49ff0b761faa30b36c..054810659af2b05f7a76455ff792eea067426519 100644 --- a/src/messaging/service.ts +++ b/src/messaging/service.ts @@ -10,6 +10,9 @@ import { ConfirmDialogComponent } from "src/components/confirmDialog/confirmDial import { IMessagingActions, IMessagingActionTmpl, WINDOW_MESSAGING_HANDLER_TOKEN, IWindowMessaging } from './types' import { TYPE as NMV_TYPE, processJsonLd as nmvProcess } from './nmvSwc/index' import { TYPE as NATIVE_TYPE, processJsonLd as nativeProcess } from './native' +import { BoothVisitor, JRPCRequest, ListenerChannel } from "src/api/jsonrpc" +import { ApiService } from "src/api"; +import { ApiBoothEvents } from "src/api/service"; export const IAV_POSTMESSAGE_NAMESPACE = `ebrains:iav:` @@ -18,12 +21,22 @@ export const MANAGED_METHODS = [ 'openminds:nmv:unloadSwc' ] +class WindowOpenerListener implements ListenerChannel { + constructor( + public registerLeaveCb: () => void, + public notify: (payload: JRPCRequest<unknown, unknown>) => void + ){} + +} + @Injectable({ providedIn: 'root' }) export class MessagingService { + private originListenerMap = new Map<string, {listener: WindowOpenerListener, visitor: BoothVisitor<ApiBoothEvents>}>() + private whiteListedOrigins = new Set() private pendingRequests: Map<string, Promise<boolean>> = new Map() private windowName: string @@ -34,6 +47,7 @@ export class MessagingService { private dialog: MatDialog, private snackbar: MatSnackBar, private worker: AtlasWorkerService, + private apiService: ApiService, @Optional() @Inject(WINDOW_MESSAGING_HANDLER_TOKEN) private messagingHandler: IWindowMessaging, ){ @@ -64,7 +78,27 @@ export class MessagingService { const src = source as Window const { id } = data try { - const result = await this.handleMessage({ data, origin }) + let result = await this.handleMessage({ data, origin }) + if (!this.originListenerMap.has(origin)) { + const listener = new WindowOpenerListener(() => { + this.apiService.broadcastCh.listeners + }, val => src.postMessage(val, origin)) + + const visitor = this.apiService.booth.handshake() + this.originListenerMap.set(origin, {listener, visitor}) + + this.apiService.broadcastCh.addListener(listener) + + + /** + * if result was not yet populated, try populating it with + * siibra-explorer api + */ + } + if (!result) { + const { visitor } = this.originListenerMap.get(origin) + return await visitor.request(data) + } src.postMessage({ id, jsonrpc: '2.0', @@ -94,27 +128,25 @@ export class MessagingService { public async handleMessage({ data, origin }) { const { method, param } = data - - if (!method) return - if (method.indexOf(IAV_POSTMESSAGE_NAMESPACE) !== 0) return - const strippedMethod = method.replace(IAV_POSTMESSAGE_NAMESPACE, '') - /** * if ping method, respond pong method */ - if (strippedMethod === 'ping') { + if (method === 'ping') { return 'pong' } /** * otherwise, check permission */ - const allow = await this.checkOrigin({ origin }) if (!allow) throw ({ code: 403, message: 'User declined' }) + + if (!method) return + if (method.indexOf(IAV_POSTMESSAGE_NAMESPACE) !== 0) return + const strippedMethod = method.replace(IAV_POSTMESSAGE_NAMESPACE, '') // TODO // in future, check if in managed_methods @@ -212,7 +244,7 @@ export class MessagingService { return await this.processJsonld(param) } - throw ({ code: 404, message: 'Method not found' }) + return } async checkOrigin({ origin }): Promise<boolean> { diff --git a/src/viewerModule/nehuba/layoutOverlay/nehuba.layoutOverlay/nehuba.layoutOverlay.component.ts b/src/viewerModule/nehuba/layoutOverlay/nehuba.layoutOverlay/nehuba.layoutOverlay.component.ts index a6e58b8462cb398d87f2f6b26b61607110950f2d..118031cc0806bb692dead3bff0f7811adae32759 100644 --- a/src/viewerModule/nehuba/layoutOverlay/nehuba.layoutOverlay/nehuba.layoutOverlay.component.ts +++ b/src/viewerModule/nehuba/layoutOverlay/nehuba.layoutOverlay/nehuba.layoutOverlay.component.ts @@ -4,7 +4,7 @@ import { combineLatest, fromEvent, interval, merge, Observable, of, Subject, Sub import { userInterface } from "src/state"; import { NehubaViewerUnit } from "../../nehubaViewer/nehubaViewer.component"; import { NEHUBA_INSTANCE_INJTKN, takeOnePipe, getFourPanel, getHorizontalOneThree, getSinglePanel, getPipPanel, getVerticalOneThree } from "../../util"; -import { QUICKTOUR_DESC, ARIA_LABELS, IDS } from 'common/constants' +import { QUICKTOUR_DESC, QUICKTOUR_DESC_MD, ARIA_LABELS, IDS } from 'common/constants' import { IQuickTourData } from "src/ui/quickTour/constrants"; import { debounce, debounceTime, distinctUntilChanged, filter, map, mapTo, switchMap, take } from "rxjs/operators"; import {panelOrder} from "src/state/userInterface/selectors"; @@ -28,6 +28,7 @@ export class NehubaLayoutOverlay implements OnDestroy{ public quickTourSliceViewSlide: IQuickTourData = { order: 1, description: QUICKTOUR_DESC.SLICE_VIEW, + descriptionMd: QUICKTOUR_DESC_MD.SLICE_VIEW } public quickTour3dViewSlide: IQuickTourData = { diff --git a/src/viewerModule/nehuba/layoutOverlay/nehuba.layoutOverlay/nehuba.layoutOverlay.template.html b/src/viewerModule/nehuba/layoutOverlay/nehuba.layoutOverlay/nehuba.layoutOverlay.template.html index 5ae334a2db080f80a983734c1004c3e3e46e231b..18968039f924b3625f13254ff2b2d6e0bf346f61 100644 --- a/src/viewerModule/nehuba/layoutOverlay/nehuba.layoutOverlay/nehuba.layoutOverlay.template.html +++ b/src/viewerModule/nehuba/layoutOverlay/nehuba.layoutOverlay/nehuba.layoutOverlay.template.html @@ -7,6 +7,7 @@ [iav-window-resize-time]="64" (iav-window-resize-event)="setQuickTourPos()" quick-tour + [quick-tour-description-md]="quickTourSliceViewSlide.descriptionMd" [quick-tour-description]="quickTourSliceViewSlide.description" [quick-tour-order]="quickTourSliceViewSlide.order" [quick-tour-overwrite-arrow]="sliceViewArrow"