import OHIF, { errorHandler } from '@ohif/core'; import React from 'react'; import * as cornerstone from '@cornerstonejs/core'; import * as cornerstoneTools from '@cornerstonejs/tools'; import { init as cs3DInit, eventTarget, EVENTS, metaData, volumeLoader, imageLoadPoolManager, getEnabledElement, Settings, utilities as csUtilities, } from '@cornerstonejs/core'; import { cornerstoneStreamingImageVolumeLoader, cornerstoneStreamingDynamicImageVolumeLoader, } from '@cornerstonejs/core/loaders'; import RequestTypes from '@cornerstonejs/core/enums/RequestType'; import initWADOImageLoader from './initWADOImageLoader'; import initCornerstoneTools from './initCornerstoneTools'; import { connectToolsToMeasurementService } from './initMeasurementService'; import initCineService from './initCineService'; import initStudyPrefetcherService from './initStudyPrefetcherService'; import interleaveCenterLoader from './utils/interleaveCenterLoader'; import nthLoader from './utils/nthLoader'; import interleaveTopToBottom from './utils/interleaveTopToBottom'; import initContextMenu from './initContextMenu'; import initDoubleClick from './initDoubleClick'; import initViewTiming from './utils/initViewTiming'; import { colormaps } from './utils/colormaps'; import { SegmentationRepresentations } from '@cornerstonejs/tools/enums'; import { useLutPresentationStore } from './stores/useLutPresentationStore'; import { usePositionPresentationStore } from './stores/usePositionPresentationStore'; import { useSegmentationPresentationStore } from './stores/useSegmentationPresentationStore'; const { registerColormap } = csUtilities.colormap; // TODO: Cypress tests are currently grabbing this from the window? (window as any).cornerstone = cornerstone; (window as any).cornerstoneTools = cornerstoneTools; /** * */ export default async function init({ servicesManager, commandsManager, extensionManager, appConfig, }: withAppTypes): Promise { // Note: this should run first before initializing the cornerstone // DO NOT CHANGE THE ORDER await cs3DInit({ peerImport: appConfig.peerImport, }); // For debugging e2e tests that are failing on CI cornerstone.setUseCPURendering(Boolean(appConfig.useCPURendering)); cornerstone.setConfiguration({ ...cornerstone.getConfiguration(), rendering: { ...cornerstone.getConfiguration().rendering, strictZSpacingForVolumeViewport: appConfig.strictZSpacingForVolumeViewport, }, }); // For debugging large datasets, otherwise prefer the defaults const { maxCacheSize } = appConfig; if (maxCacheSize) { cornerstone.cache.setMaxCacheSize(maxCacheSize); } initCornerstoneTools(); Settings.getRuntimeSettings().set('useCursors', Boolean(appConfig.useCursors)); const { userAuthenticationService, customizationService, uiModalService, uiNotificationService, cornerstoneViewportService, hangingProtocolService, viewportGridService, } = servicesManager.services; window.services = servicesManager.services; window.extensionManager = extensionManager; window.commandsManager = commandsManager; if (appConfig.showCPUFallbackMessage && cornerstone.getShouldUseCPURendering()) { _showCPURenderingModal(uiModalService, hangingProtocolService); } const { getPresentationId: getLutPresentationId } = useLutPresentationStore.getState(); const { getPresentationId: getSegmentationPresentationId } = useSegmentationPresentationStore.getState(); const { getPresentationId: getPositionPresentationId } = usePositionPresentationStore.getState(); // register presentation id providers viewportGridService.addPresentationIdProvider( 'positionPresentationId', getPositionPresentationId ); viewportGridService.addPresentationIdProvider('lutPresentationId', getLutPresentationId); viewportGridService.addPresentationIdProvider( 'segmentationPresentationId', getSegmentationPresentationId ); cornerstoneTools.segmentation.config.style.setStyle( { type: SegmentationRepresentations.Contour }, { renderFill: false, } ); const metadataProvider = OHIF.classes.MetadataProvider; volumeLoader.registerVolumeLoader( 'cornerstoneStreamingImageVolume', cornerstoneStreamingImageVolumeLoader ); volumeLoader.registerVolumeLoader( 'cornerstoneStreamingDynamicImageVolume', cornerstoneStreamingDynamicImageVolumeLoader ); hangingProtocolService.registerImageLoadStrategy('interleaveCenter', interleaveCenterLoader); hangingProtocolService.registerImageLoadStrategy('interleaveTopToBottom', interleaveTopToBottom); hangingProtocolService.registerImageLoadStrategy('nth', nthLoader); // add metadata providers metaData.addProvider( csUtilities.calibratedPixelSpacingMetadataProvider.get.bind( csUtilities.calibratedPixelSpacingMetadataProvider ) ); // this provider is required for Calibration tool metaData.addProvider(metadataProvider.get.bind(metadataProvider), 9999); // These are set reasonably low to allow for interleaved retrieves and slower // connections. imageLoadPoolManager.maxNumRequests = { [RequestTypes.Interaction]: appConfig?.maxNumRequests?.interaction || 10, [RequestTypes.Thumbnail]: appConfig?.maxNumRequests?.thumbnail || 5, [RequestTypes.Prefetch]: appConfig?.maxNumRequests?.prefetch || 5, [RequestTypes.Compute]: appConfig?.maxNumRequests?.compute || 10, }; initWADOImageLoader(userAuthenticationService, appConfig, extensionManager); /* Measurement Service */ this.measurementServiceSource = connectToolsToMeasurementService(servicesManager); initCineService(servicesManager); initStudyPrefetcherService(servicesManager); // When a custom image load is performed, update the relevant viewports hangingProtocolService.subscribe( hangingProtocolService.EVENTS.CUSTOM_IMAGE_LOAD_PERFORMED, volumeInputArrayMap => { const { lutPresentationStore } = useLutPresentationStore.getState(); const { segmentationPresentationStore } = useSegmentationPresentationStore.getState(); const { positionPresentationStore } = usePositionPresentationStore.getState(); for (const entry of volumeInputArrayMap.entries()) { const [viewportId, volumeInputArray] = entry; const viewport = cornerstoneViewportService.getCornerstoneViewport(viewportId); const ohifViewport = cornerstoneViewportService.getViewportInfo(viewportId); const { presentationIds } = ohifViewport.getViewportOptions(); const presentations = { positionPresentation: positionPresentationStore[presentationIds?.positionPresentationId], lutPresentation: lutPresentationStore[presentationIds?.lutPresentationId], segmentationPresentation: segmentationPresentationStore[presentationIds?.segmentationPresentationId], }; cornerstoneViewportService.setVolumesForViewport(viewport, volumeInputArray, presentations); } } ); // resize the cornerstone viewport service when the grid size changes // IMPORTANT: this should happen outside of the OHIFCornerstoneViewport // since it will trigger a rerender of each viewport and each resizing // the offscreen canvas which would result in a performance hit, this should // done only once per grid resize here. Doing it once here, allows us to reduce // the refreshRage(in ms) to 10 from 50. I tried with even 1 or 5 ms it worked fine viewportGridService.subscribe(viewportGridService.EVENTS.GRID_SIZE_CHANGED, () => { cornerstoneViewportService.resize(true); }); initContextMenu({ cornerstoneViewportService, customizationService, commandsManager, }); initDoubleClick({ customizationService, commandsManager, }); /** * Runs error handler for failed requests. * @param event */ const imageLoadFailedHandler = ({ detail }) => { const handler = errorHandler.getHTTPErrorHandler(); handler(detail.error); }; eventTarget.addEventListener(EVENTS.IMAGE_LOAD_FAILED, imageLoadFailedHandler); eventTarget.addEventListener(EVENTS.IMAGE_LOAD_ERROR, imageLoadFailedHandler); function elementEnabledHandler(evt) { const { element } = evt.detail; element.addEventListener(EVENTS.CAMERA_RESET, evt => { const { element } = evt.detail; const enabledElement = getEnabledElement(element); if (!enabledElement) { return; } const { viewportId } = enabledElement; commandsManager.runCommand('resetCrosshairs', { viewportId }); }); initViewTiming({ element }); } eventTarget.addEventListener(EVENTS.ELEMENT_ENABLED, elementEnabledHandler.bind(null)); colormaps.forEach(registerColormap); // Event listener eventTarget.addEventListenerDebounced( EVENTS.ERROR_EVENT, ({ detail }) => { uiNotificationService.show({ title: detail.type, message: detail.message, type: 'error', }); }, 100 ); // Call this function when initializing initializeWebWorkerProgressHandler(servicesManager.services.uiNotificationService); } function initializeWebWorkerProgressHandler(uiNotificationService) { const activeToasts = new Map(); eventTarget.addEventListener(EVENTS.WEB_WORKER_PROGRESS, ({ detail }) => { const { progress, type, id } = detail; const cacheKey = `${type}-${id}`; if (progress === 0 && !activeToasts.has(cacheKey)) { const progressPromise = new Promise((resolve, reject) => { activeToasts.set(cacheKey, { resolve, reject }); }); uiNotificationService.show({ id: cacheKey, title: `${type}`, message: `${type}: ${progress}%`, autoClose: false, promise: progressPromise, promiseMessages: { loading: `Computing...`, success: `Completed successfully`, error: 'Web Worker failed', }, }); } else { if (progress === 100) { const { resolve } = activeToasts.get(cacheKey); resolve({ progress, type }); activeToasts.delete(cacheKey); } } }); } function CPUModal() { return (

Your computer does not have enough GPU power to support the default GPU rendering mode. OHIF has switched to CPU rendering mode. Please note that CPU rendering does not support all features such as Volume Rendering, Multiplanar Reconstruction, and Segmentation Overlays.

); } function _showCPURenderingModal(uiModalService, hangingProtocolService) { const callback = progress => { if (progress === 100) { uiModalService.show({ content: CPUModal, title: 'OHIF Fell Back to CPU Rendering', }); return true; } }; const { unsubscribe } = hangingProtocolService.subscribe( hangingProtocolService.EVENTS.PROTOCOL_CHANGED, () => { const done = callback(100); if (done) { unsubscribe(); } } ); }