Newer
Older
import { Component, OnDestroy, Output, EventEmitter, ElementRef, NgZone, Renderer2, OnInit } from "@angular/core";
import { fromEvent, Subscription, Subject } from 'rxjs'
import { AtlasWorkerService } from "src/atlasViewer/atlasViewer.workerService.service";
import { map, filter, debounceTime, scan } from "rxjs/operators";
import { AtlasViewerConstantsServices } from "src/atlasViewer/atlasViewer.constantService.service";
import { takeOnePipe } from "../nehubaContainer.component";
import { ViewerConfiguration } from "src/services/state/viewerConfig.store";
import { pipeFromArray } from "rxjs/internal/util/pipe";
import { getNgIdLabelIndexFromId } from "src/services/stateStore.service";
import 'third_party/export_nehuba/main.bundle.js'
import 'third_party/export_nehuba/chunk_worker.bundle.js'
interface LayerLabelIndex {
layer: {
name: string
}
labelIndicies: number[]
}
const scanFn : (acc: LayerLabelIndex[], curr: LayerLabelIndex) => LayerLabelIndex[] = (acc: LayerLabelIndex[], curr: LayerLabelIndex) => {
const { layer } = curr
const { name } = layer
const foundIndex = acc.findIndex(({ layer }) => layer.name === name)
if (foundIndex < 0) return acc.concat(curr)
else return acc.map((item, idx) => idx === foundIndex
? {
...item,
labelIndicies: [...new Set([...item.labelIndicies, ...curr.labelIndicies])]
}
: item)
}
/**
* no selector is needed, as currently, nehubaviewer is created dynamically
*/
@Component({
templateUrl : './nehubaViewer.template.html',
styleUrls : [
'./nehubaViewer.style.css'
]
})
export class NehubaViewerUnit implements OnInit, OnDestroy{
private subscriptions: Subscription[] = []
@Output() nehubaReady: EventEmitter<null> = new EventEmitter()
@Output() layersChanged: EventEmitter<null> = new EventEmitter()
private layersChangedHandler: any
@Output() debouncedViewerPositionChange : EventEmitter<any> = new EventEmitter()
@Output() mouseoverSegmentEmitter:
EventEmitter<{
segmentId: number | null,
segment:string | null,
layer:{
name?: string,
url?: string
}
}> = new EventEmitter()
@Output() mouseoverLandmarkEmitter : EventEmitter<number | null> = new EventEmitter()
@Output() mouseoverUserlandmarkEmitter: EventEmitter<number | null> = new EventEmitter()
@Output() regionSelectionEmitter : EventEmitter<{segment:number, layer:{name?: string, url?: string}}> = new EventEmitter()
@Output() errorEmitter : EventEmitter<any> = new EventEmitter()
/* only used to set initial navigation state */
initNav : any
initRegions : any[]
initNiftiLayers : any[] = []
config : any
nehubaViewer : any
private _dim: [number, number, number]
get dim(){
return this._dim
? this._dim
: [1.5e9, 1.5e9, 1.5e9]
}
Xiao Gui
committed
_s1$ : any
_s2$ : any
_s3$ : any
_s4$ : any
_s5$ : any
_s6$ : any
_s7$ : any
Xiao Gui
committed
_s$ : any[] = [
this._s1$,
this._s2$,
this._s3$,
this._s4$,
this._s5$,
this._s6$,
Xiao Gui
committed
]
ondestroySubscriptions: Subscription[] = []
constructor(
public elementRef:ElementRef,
private workerService : AtlasWorkerService,
private zone : NgZone,
public constantService : AtlasViewerConstantsServices
if(!this.constantService.loadExportNehubaPromise){
this.constantService.loadExportNehubaPromise = new Promise((resolve, reject) => {
const scriptEl = this.rd.createElement('script')
scriptEl.src = 'main.bundle.js'
scriptEl.onload = () => resolve(true)
scriptEl.onerror = (e) => reject(e)
this.rd.appendChild(window.document.head, scriptEl)
})
}
this.constantService.loadExportNehubaPromise
.then(() => {
const fixedZoomPerspectiveSlices = this.config && this.config.layout && this.config.layout.useNehubaPerspective && this.config.layout.useNehubaPerspective.fixedZoomPerspectiveSlices
if (fixedZoomPerspectiveSlices) {
const { sliceZoom, sliceViewportWidth, sliceViewportHeight } = fixedZoomPerspectiveSlices
const dim = Math.min(sliceZoom * sliceViewportWidth, sliceZoom * sliceViewportHeight)
this._dim = [dim, dim, dim]
}
this.patchNG()
this.loadNehuba()
this.layersChangedHandler = this.nehubaViewer.ngviewer.layerManager.layersChanged.add(() => this.layersChanged.emit(null))
this.nehubaViewer.ngviewer.registerDisposer(this.layersChangedHandler)
})
.catch(e => this.errorEmitter.emit(e))
this.ondestroySubscriptions.push(
fromEvent(this.workerService.worker, 'message').pipe(
filter((message:any) => {
if(!message){
// console.error('worker response message is undefined', message)
return false
}
if(!message.data){
// console.error('worker response message.data is undefined', message.data)
return false
}
if(message.data.type !== 'ASSEMBLED_LANDMARKS_VTK'){
/* worker responded with not assembled landmark, no need to act */
return false
}
if(!message.data.url){
/* file url needs to be defined */
return false
}
return true
}),
debounceTime(100),
filter(e => this.templateId === e.data.template),
map(e => e.data.url)
).subscribe(url => {
this.removeSpatialSearch3DLandmarks()
const _ = {}
_[this.constantService.ngLandmarkLayerName] = {
type :'mesh',
source : `vtk://${url}`,
shader : FRAGMENT_MAIN_WHITE
}
this.loadLayer(_)
})
)
this.ondestroySubscriptions.push(
fromEvent(this.workerService.worker, 'message').pipe(
filter((message:any) => {
if(!message){
// console.error('worker response message is undefined', message)
return false
}
if(!message.data){
// console.error('worker response message.data is undefined', message.data)
return false
}
if(message.data.type !== 'ASSEMBLED_USERLANDMARKS_VTK'){
/* worker responded with not assembled landmark, no need to act */
return false
}
/**
* nb url may be undefined
* if undefined, user have removed all user landmarks, and all that needs to be done
* is remove the user landmark layer
*
* message.data.url
*/
return true
}),
debounceTime(100),
map(e => e.data.url)
).subscribe(url => {
this.removeuserLandmarks()
/**
* url may be null if user removes all landmarks
*/
if (!url) return
const _ = {}
_[this.constantService.ngUserLandmarkLayerName] = {
type :'mesh',
source : `vtk://${url}`,
shader : FRAGMENT_MAIN_WHITE
}
this.loadLayer(_)
})
)
private _baseUrlToParcellationIdMap:Map<string, string> = new Map()
private _baseUrls: string[] = []
public numMeshesToBeLoaded: number = 0
public applyPerformanceConfig ({ gpuLimit }: Partial<ViewerConfiguration>) {
if (gpuLimit && this.nehubaViewer) {
const limit = this.nehubaViewer.ngviewer.state.children.get('gpuMemoryLimit')
if (limit && limit.restoreState) {
limit.restoreState(gpuLimit)
}
}
}
/* required to check if correct landmarks are loaded */
private _templateId : string
get templateId(){
return this._templateId
}
set templateId(id:string){
this._templateId = id
}
/** compatible with multiple parcellation id selection */
private _ngIds: string[] = []
get ngIds(){
return this._ngIds
set ngIds(val:string[]){
if(this.nehubaViewer){
this._ngIds.forEach(id => {
const oldlayer = this.nehubaViewer.ngviewer.layerManager.getLayerByName(id)
if(oldlayer)oldlayer.setVisible(false)
else console.warn('could not find old layer', id)
})
spatialLandmarkSelectionChanged(labels:number[]){
const getCondition = (label:number) => `if(label > ${label - 0.1} && label < ${label + 0.1} ){${FRAGMENT_EMIT_RED}}`
const newShader = `void main(){ ${labels.map(getCondition).join('else ')}else {${FRAGMENT_EMIT_WHITE}} }`
if(!this.nehubaViewer){
if(!PRODUCTION || window['__debug__']) console.warn('setting special landmark selection changed failed ... nehubaViewer is not yet defined')
return
}
const landmarkLayer = this.nehubaViewer.ngviewer.layerManager.getLayerByName(this.constantService.ngLandmarkLayerName)
if(!landmarkLayer){
if(!PRODUCTION || window['__debug__']) console.warn('landmark layer could not be found ... will not update colour map')
return
}
if(labels.length === 0){
landmarkLayer.layer.displayState.fragmentMain.restoreState(FRAGMENT_MAIN_WHITE)
}else{
landmarkLayer.layer.displayState.fragmentMain.restoreState(newShader)
}
}
multiNgIdsLabelIndexMap: Map<string, Map<number, any>>
Xiao Gui
committed
navPosReal : [number,number,number] = [0,0,0]
navPosVoxel : [number,number,number] = [0,0,0]
mousePosReal : [number,number,number] = [0,0,0]
mousePosVoxel : [number,number,number] = [0,0,0]
viewerState : ViewerState
private _multiNgIdColorMap: Map<string, Map<number, {red: number, green:number, blue: number}>>
get multiNgIdColorMap(){
return this._multiNgIdColorMap
}
set multiNgIdColorMap(val) {
this._multiNgIdColorMap = val
}
private loadMeshes$: Subject<{labelIndicies: number[], layer: { name: string }}> = new Subject()
private loadMeshes(labelIndicies: number[], { name }){
this.loadMeshes$.next({
labelIndicies,
layer: { name }
})
}
public mouseOverSegment: number | null
public mouseOverLayer: {name:string,url:string}| null
public viewportToDatas : [any, any, any] = [null, null, null]
public getNgHash : () => string = () => window['export_nehuba']
? window['export_nehuba'].getNgHash()
: null
redraw(){
this.nehubaViewer.redraw()
}
loadNehuba(){
this.nehubaViewer = window['export_nehuba'].createNehubaViewer(this.config, (err)=>{
/* print in debug mode */
/**
* Hide all layers except the base layer (template)
* Then show the layers referenced in multiNgIdLabelIndexMap
*/
const managedLayers = this.nehubaViewer.ngviewer.layerManager.managedLayers
managedLayers.slice(1).forEach(layer=>layer.setVisible(false))
Array.from(this.multiNgIdsLabelIndexMap.keys()).forEach(ngId => {
const layer = this.nehubaViewer.ngviewer.layerManager.getLayerByName(ngId)
if (layer) layer.setVisible(true)
else console.log('layer unavailable', ngId)
})
Xiao Gui
committed
/* creation of the layout is done on next frame, hence the settimeout */
setTimeout(() => {
window['viewer'].display.panels.forEach(patchSliceViewPanel)
this.nehubaReady.emit(null)
this.newViewerInit()
this.loadNewParcellation()
Xiao Gui
committed
window['nehubaViewer'] = this.nehubaViewer
this.onDestroyCb.push(() => window['nehubaViewer'] = null)
/**
* TODO
* move this to nehubaContainer
* do NOT use position logic to determine idx
*/
// fromEvent(this.elementRef.nativeElement, 'viewportToData').pipe(
// ...takeOnePipe
// ).subscribe((events:CustomEvent[]) => {
// [0,1,2].forEach(idx => this.viewportToDatas[idx] = events[idx].detail.viewportToData)
// })
pipeFromArray([...takeOnePipe])(fromEvent(this.elementRef.nativeElement, 'viewportToData'))
.subscribe((events:CustomEvent[]) => {
[0,1,2].forEach(idx => this.viewportToDatas[idx] = events[idx].detail.viewportToData)
})
ngOnInit(){
this.subscriptions.push(
this.loadMeshes$.pipe(
scan(scanFn, [])
).subscribe(layersLabelIndex => {
let totalMeshes = 0
for (const layerLayerIndex of layersLabelIndex){
const { layer, labelIndicies } = layerLayerIndex
totalMeshes += labelIndicies.length
this.nehubaViewer.setMeshesToLoad(labelIndicies, layer)
}
// TODO implement total mesh to be loaded and mesh loading UI
// this.numMeshesToBeLoaded = totalMeshes
})
)
}
while(this.subscriptions.length > 0) {
this.subscriptions.pop().unsubscribe()
}
Xiao Gui
committed
this._s$.forEach(_s$=>{
if(_s$) _s$.unsubscribe()
})
this.ondestroySubscriptions.forEach(s => s.unsubscribe())
while(this.onDestroyCb.length > 0){
this.onDestroyCb.pop()()
}
private onDestroyCb : (()=>void)[] = []
const { LayerManager, UrlHashBinding } = window['export_nehuba'].getNgPatchableObj()
UrlHashBinding.prototype.setUrlHash = () => {
// console.log('seturl hash')
// console.log('setting url hash')
}
UrlHashBinding.prototype.updateFromUrlHash = () => {
// console.log('update hash binding')
}
/* TODO find a more permanent fix to disable double click */
LayerManager.prototype.invokeAction = (arg) => {
/**
* The emitted value does not affect the region selection
* the region selection is taken care of in nehubaContainer
*/
const map = this.multiNgIdsLabelIndexMap.get(this.mouseOverLayer.name)
const region = map && map.get(this.mouseOverSegment)
if (arg === 'select') {
this.regionSelectionEmitter.emit({ segment: region, layer: this.mouseOverLayer })
this.onDestroyCb.push(() => LayerManager.prototype.invokeAction = (arg) => {})
private filterLayers(l:any,layerObj:any):boolean{
/**
* if selector is an empty object, select all layers
*/
return layerObj instanceof Object && Object.keys(layerObj).every(key =>
/**
* the property described by the selector must exist and ...
*/
!!l[key] &&
/**
* if the selector is regex, test layer property
*/
( layerObj[key] instanceof RegExp
? layerObj[key].test(l[key])
/**
* if selector is string, test for strict equality
*/
: typeof layerObj[key] === 'string'
? layerObj[key] === l[key]
/**
* otherwise do not filter
*/
: false
)
)
if(!this.nehubaViewer)
return
this.workerService.worker.postMessage({
type : 'GET_USERLANDMARKS_VTK',
scale: Math.min(...this.dim.map(v => v * this.constantService.nehubaLandmarkConstant)),
landmarks : landmarks.map(lm => lm.position.map(coord => coord * 1e6))
})
}
public removeSpatialSearch3DLandmarks(){
this.removeLayer({
name : this.constantService.ngLandmarkLayerName
})
}
this.removeLayer({
name : this.constantService.ngUserLandmarkLayerName
})
}
//pos in mm
public addSpatialSearch3DLandmarks(geometries: any[],scale?:number,type?:'icosahedron'){
this.workerService.worker.postMessage({
type : 'GET_LANDMARKS_VTK',
template : this.templateId,
scale: Math.min(...this.dim.map(v => v * this.constantService.nehubaLandmarkConstant)),
landmarks : geometries.map(geometry =>
geometry === null
? null
//gemoetry : [number,number,number] | [ [number,number,number][], [number,number,number][] ]
: isNaN(geometry[0])
? [geometry[0].map(triplets => triplets.map(coord => coord * 1e6)), geometry[1]]
: geometry.map(coord => coord * 1e6)
)
public setLayerVisibility(condition:{name:string|RegExp},visible:boolean){
if(!this.nehubaViewer)
return false
const viewer = this.nehubaViewer.ngviewer
viewer.layerManager.managedLayers
.filter(l=>this.filterLayers(l,condition))
.map(layer=>layer.setVisible(visible))
}
if(!this.nehubaViewer)
return false
const viewer = this.nehubaViewer.ngviewer
const removeLayer = (i) => (viewer.layerManager.removeManagedLayer(i),i.name)
return viewer.layerManager.managedLayers
.filter(l=>this.filterLayers(l,layerObj))
.map(removeLayer)
}
public loadLayer(layerObj:any){
const viewer = this.nehubaViewer.ngviewer
return Object.keys(layerObj)
.filter(key=>
/* if the layer exists, it will not be loaded */
!viewer.layerManager.getLayerByName(key))
.map(key=>{
viewer.layerManager.addManagedLayer(
viewer.layerSpecification.getLayer(key,layerObj[key]))
return layerObj[key]
})
}
Array.from(this.multiNgIdsLabelIndexMap.keys()).forEach(ngId => {
Array.from(this.multiNgIdsLabelIndexMap.get(ngId).keys()).forEach(idx => {
this.nehubaViewer.hideSegment(idx, {
name: ngId
})
})
this.nehubaViewer.showSegment(0, {
name: ngId
})
})
}
public showAllSeg(){
Xiao Gui
committed
this.hideAllSeg()
Array.from(this.multiNgIdsLabelIndexMap.keys()).forEach(ngId => {
this.nehubaViewer.hideSegment(0,{
name: ngId
})
Xiao Gui
committed
})
public showSegs(array: number[] | string[]){
Xiao Gui
committed
this.hideAllSeg()
if (array.length === 0) return
Xiao Gui
committed
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
/**
* TODO tobe deprecated
*/
if (typeof array[0] === 'number') {
console.warn(`show seg with number indices has been deprecated`)
return
}
const reduceFn:(acc:Map<string,number[]>,curr:string)=>Map<string,number[]> = (acc, curr) => {
const newMap = new Map(acc)
const { ngId, labelIndex } = getNgIdLabelIndexFromId({ labelIndexId: curr })
const exist = newMap.get(ngId)
if (!exist) newMap.set(ngId, [Number(labelIndex)])
else newMap.set(ngId, [...exist, Number(labelIndex)])
return newMap
}
/**
* TODO
* AAAAAAARG TYPESCRIPT WHY YOU SO MAD
*/
//@ts-ignore
const newMap:Map<string, number[]> = array.reduce(reduceFn, new Map())
/**
* TODO
* ugh, ugly code. cleanify
*/
/**
* TODO
* sometimes, ngId still happends to be undefined
*/
newMap.forEach((segs, ngId) => {
this.nehubaViewer.hideSegment(0, {
name: ngId
})
segs.forEach(seg => {
this.nehubaViewer.showSegment(seg, {
name: ngId
})
})
})
private vec3(pos:[number,number,number]){
return window['export_nehuba'].vec3.fromValues(...pos)
Xiao Gui
committed
public setNavigationState(newViewerState:Partial<ViewerState>){
if(!PRODUCTION || window['__debug__'])
console.warn('setNavigationState > this.nehubaViewer is not yet defined')
Xiao Gui
committed
const {
orientation,
perspectiveOrientation,
perspectiveZoom,
position,
positionReal,
zoom
} = newViewerState
if( perspectiveZoom )
this.nehubaViewer.ngviewer.perspectiveNavigationState.zoomFactor.restoreState(perspectiveZoom)
if( zoom )
this.nehubaViewer.ngviewer.navigationState.zoomFactor.restoreState(zoom)
if( perspectiveOrientation )
this.nehubaViewer.ngviewer.perspectiveNavigationState.pose.orientation.restoreState( perspectiveOrientation )
if( orientation )
this.nehubaViewer.ngviewer.navigationState.pose.orientation.restoreState( orientation )
if( position )
this.nehubaViewer.setPosition( this.vec3(position) , positionReal ? true : false )
Xiao Gui
committed
}
public obliqueRotateX(amount:number){
this.nehubaViewer.ngviewer.navigationState.pose.rotateRelative(this.vec3([0, 1, 0]), -amount / 4.0 * Math.PI / 180.0)
}
public obliqueRotateY(amount:number){
this.nehubaViewer.ngviewer.navigationState.pose.rotateRelative(this.vec3([1, 0, 0]), amount / 4.0 * Math.PI / 180.0)
}
public obliqueRotateZ(amount:number){
this.nehubaViewer.ngviewer.navigationState.pose.rotateRelative(this.vec3([0, 0, 1]), amount / 4.0 * Math.PI / 180.0)
}
/**
*
* @param arrayIdx label indices of the shown segment(s)
* @param ngId segmentation layer name
*/
private updateColorMap(arrayIdx:number[], ngId: string){
const set = new Set(arrayIdx)
const newColorMap = new Map(
Array.from(this.multiNgIdColorMap.get(ngId).entries())
.map(v=> set.has(v[0]) || set.size === 0 ?
v :
[v[0],{red:255,green:255,blue:255}]) as any
)
this.nehubaViewer.batchAddAndUpdateSegmentColors(newColorMap,{
/* isn't this layer specific? */
/* TODO this is layer specific. need a way to distinguish between different segmentation layers */
this._s2$ = this.nehubaViewer.mouseOver.segment
.subscribe(({ segment, layer })=>{
this.mouseOverSegment = segment
this.mouseOverLayer = { ...layer }
})
if(this.initNav){
this.setNavigationState(this.initNav)
}
Xiao Gui
committed
if(this.initRegions){
this.hideAllSeg()
this.showSegs(this.initRegions)
}
if(this.initNiftiLayers.length > 0){
this.initNiftiLayers.forEach(layer => this.loadLayer(layer))
this.hideAllSeg()
}
this._s8$ = this.nehubaViewer.mouseOver.segment.subscribe(({segment: segmentId, layer, ...rest})=>{
const {name = 'unnamed'} = layer
const map = this.multiNgIdsLabelIndexMap.get(name)
const region = map && map.get(segmentId)
this.mouseoverSegmentEmitter.emit({
layer,
segment: region,
segmentId
})
Xiao Gui
committed
// nehubaViewer.navigationState.all emits every time a new layer is added or removed from the viewer
Xiao Gui
committed
this._s3$ = this.nehubaViewer.navigationState.all
.debounceTime(300)
.distinctUntilChanged((a, b) => {
const {
orientation: o1,
perspectiveOrientation: po1,
perspectiveZoom: pz1,
position: p1,
zoom: z1
} = a
const {
orientation: o2,
perspectiveOrientation: po2,
perspectiveZoom: pz2,
position: p2,
zoom: z2
} = b
return [0,1,2,3].every(idx => o1[idx] === o2[idx]) &&
[0,1,2,3].every(idx => po1[idx] === po2[idx]) &&
pz1 === pz2 &&
[0,1,2].every(idx => p1[idx] === p2[idx]) &&
z1 === z2
})
Xiao Gui
committed
.subscribe(({ orientation, perspectiveOrientation, perspectiveZoom, position, zoom })=>{
this.viewerState = {
orientation,
perspectiveOrientation,
perspectiveZoom,
zoom,
position,
positionReal : false
}
this.debouncedViewerPositionChange.emit({
orientation : Array.from(orientation),
perspectiveOrientation : Array.from(perspectiveOrientation),
perspectiveZoom,
zoom,
position,
positionReal : true
})
Xiao Gui
committed
})
this.ondestroySubscriptions.push(
this.nehubaViewer.mouseOver.layer
.filter(obj => obj.layer.name === this.constantService.ngLandmarkLayerName)
.subscribe(obj => this.mouseoverLandmarkEmitter.emit(obj.value))
)
this.ondestroySubscriptions.push(
this.nehubaViewer.mouseOver.layer
.filter(obj => obj.layer.name === this.constantService.ngUserLandmarkLayerName)
.subscribe(obj => this.mouseoverUserlandmarkEmitter.emit(obj.value))
)
Xiao Gui
committed
this._s4$ = this.nehubaViewer.navigationState.position.inRealSpace
.filter(v=>typeof v !== 'undefined' && v !== null)
.subscribe(v=>this.navPosReal=v)
this._s5$ = this.nehubaViewer.navigationState.position.inVoxels
.filter(v=>typeof v !== 'undefined' && v !== null)
.subscribe(v=>this.navPosVoxel=v)
this._s6$ = this.nehubaViewer.mousePosition.inRealSpace
.filter(v=>typeof v !== 'undefined' && v !== null)
.subscribe(v=>(this.mousePosReal=v))
this._s7$ = this.nehubaViewer.mousePosition.inVoxels
.filter(v=>typeof v !== 'undefined' && v !== null)
.subscribe(v=>(this.mousePosVoxel=v))
private loadNewParcellation(){
/* show correct segmentation layer */
this._baseUrls = []
this.ngIds.map(id => {
const newlayer = this.nehubaViewer.ngviewer.layerManager.getLayerByName(id)
if(newlayer)newlayer.setVisible(true)
else console.warn('could not find new layer',id)
const regex = /^(\S.*?)\:\/\/(.*?)$/.exec(newlayer.sourceUrl)
if(!regex || !regex[2]){
console.error('could not parse baseUrl')
return
if(regex[1] !== 'precomputed'){
console.error('sourceUrl is not precomputed')
return
}
const baseUrl = regex[2]
this._baseUrls.push(baseUrl)
this._baseUrlToParcellationIdMap.set(baseUrl, id)
const indicies = [
...Array.from(this.multiNgIdsLabelIndexMap.get(id).keys()),
...getAuxilliaryLabelIndices()
]
this.loadMeshes(indicies, { name: id })
const obj = Array.from(this.multiNgIdsLabelIndexMap.keys()).map(ngId => {
return [
ngId,
new Map(Array.from(
[
...this.multiNgIdsLabelIndexMap.get(ngId).entries(),
...getAuxilliaryLabelIndices().map(i => {
return [i, {}]
})
]
).map((val:[number,any])=>([val[0],this.getRgb(val[0],val[1].rgb)])) as any)
]
}) as [string, Map<number, {red:number, green: number, blue: number}>][]
this._s$.forEach(_s$=>{
if(_s$) _s$.unsubscribe()
})
if(this._s1$)this._s1$.unsubscribe()
if(this._s9$)this._s9$.unsubscribe()
const arr = Array.from(this.multiNgIdsLabelIndexMap.keys()).map(ngId => {
return this.nehubaViewer.getShownSegmentsObservable({
name: ngId
}).subscribe(arrayIdx => this.updateColorMap(arrayIdx, ngId))
})
this._s9$ = {
unsubscribe: () => {
while(arr.length > 0) {
arr.pop().unsubscribe()
}
}
}
public setColorMap(map: Map<string, Map<number,{red:number, green:number, blue:number}>>){
this.multiNgIdColorMap = map
Array.from(map.entries()).forEach(([ngId, map]) => {
this.nehubaViewer.batchAddAndUpdateSegmentColors(
map,
{ name : ngId })
})
}
private getRgb(labelIndex:number,rgb?:number[]):{red:number,green:number,blue:number}{
if(typeof rgb === 'undefined' || rgb === null){
const arr = intToColour(Number(labelIndex))
red : arr[0],
green: arr[1],
blue : arr[2]
return {
red : rgb[0],
green: rgb[1],
blue : rgb[2]
}
Xiao Gui
committed
}
const patchSliceViewPanel = (sliceViewPanel: any) => {
const originalDraw = sliceViewPanel.draw
sliceViewPanel.draw = function (this) {
if(this.sliceView){
const viewportToDataEv = new CustomEvent('viewportToData', {
bubbles: true,
detail: {
viewportToData : this.sliceView.viewportToData
}
})
this.element.dispatchEvent(viewportToDataEv)
}
originalDraw.call(this)
}
}
/**
*
* https://stackoverflow.com/a/16348977/6059235
*/
const intToColour = function(int) {
if(int >= 65500){
return [255,255,255]
}
const str = String(int*65535)
let hash = 0
for (let i = 0; i < str.length; i++) {
hash = str.charCodeAt(i) + ((hash << 5) - hash);
}
const returnV = []
for (let i = 0; i < 3; i++) {
const value = (hash >> (i * 8)) & 0xFF;
returnV.push(value)
}
return returnV
}
Xiao Gui
committed
export interface ViewerState{
orientation : [number,number,number,number]
perspectiveOrientation : [number,number,number,number]
perspectiveZoom : number
position : [number,number,number]
positionReal : boolean
zoom : number
}
export function getAuxilliaryLabelIndices(){
return [65535]
// return Array.from(Array(36)).map((_,i)=>65500+i)
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
}
export const ICOSAHEDRON = `# vtk DataFile Version 2.0
Converted using https://github.com/HumanBrainProject/neuroglancer-scripts
ASCII
DATASET POLYDATA
POINTS 12 float
-525731.0 0.0 850651.0
525731.0 0.0 850651.0
-525731.0 0.0 -850651.0
525731.0 0.0 -850651.0
0.0 850651.0 525731.0
0.0 850651.0 -525731.0
0.0 -850651.0 525731.0
0.0 -850651.0 -525731.0
850651.0 525731.0 0.0
-850651.0 525731.0 0.0
850651.0 -525731.0 0.0
-850651.0 -525731.0 0.0
POLYGONS 20 80
3 1 4 0
3 4 9 0
3 4 5 9
3 8 5 4
3 1 8 4
3 1 10 8
3 10 3 8
3 8 3 5
3 3 2 5
3 3 7 2
3 3 10 7
3 10 6 7
3 6 11 7
3 6 0 11
3 6 1 0
3 10 1 6
3 11 0 9
3 2 11 9
3 5 2 9
3 11 2 7`
declare const TextEncoder
export const _encoder = new TextEncoder()
export const ICOSAHEDRON_VTK_URL = URL.createObjectURL( new Blob([ _encoder.encode(ICOSAHEDRON) ],{type : 'application/octet-stream'} ))
export const FRAGMENT_MAIN_WHITE = `void main(){emitRGB(vec3(1.0,1.0,1.0));}`
export const FRAGMENT_EMIT_WHITE = `emitRGB(vec3(1.0, 1.0, 1.0));`
export const FRAGMENT_EMIT_RED = `emitRGB(vec3(1.0, 0.1, 0.12));`
export const computeDistance = (pt1:[number, number], pt2:[number,number]) => ((pt1[0] - pt2[0]) ** 2 + (pt1[1] - pt2[1]) ** 2) ** 0.5