Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix(StackSync): Miscellaneous fixes for stack image sync #3663

Merged
merged 31 commits into from
Oct 3, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
31 commits
Select commit Hold shift + click to select a range
71b4827
fix: Miscellaneous fixes for stack image sync
wayfarer3130 Sep 19, 2023
c7ae0d5
PR comments
wayfarer3130 Sep 21, 2023
270c93f
Reducing number of changed lines
wayfarer3130 Sep 21, 2023
78022dd
Reducing number of changes
wayfarer3130 Sep 21, 2023
ff63b93
Merge remote-tracking branch 'ohif/master' into fix/stack-image-sync-for
wayfarer3130 Sep 21, 2023
479ca55
PR requested changes, plus some build warning removals
wayfarer3130 Sep 22, 2023
091034c
PR review comments
wayfarer3130 Sep 22, 2023
466545c
PR review comments
wayfarer3130 Sep 22, 2023
355670c
Merge remote-tracking branch 'ohif/master' into fix/stack-image-sync-for
wayfarer3130 Sep 22, 2023
c50a9c1
Initial changes for event naming
wayfarer3130 Sep 25, 2023
96edc5a
Merge remote-tracking branch 'ohif/master' into fix/stack-image-sync-for
wayfarer3130 Sep 26, 2023
c545b89
PR requested fixes
wayfarer3130 Sep 26, 2023
62bb7bf
Merge branch 'master' into fix/stack-image-sync-for
wayfarer3130 Sep 26, 2023
e1869f9
Add tracking to show performance difference
wayfarer3130 Sep 26, 2023
0d10b6d
Merge branch 'master' of github.com:OHIF/Viewers into pr/fix/stack-im…
sedghi Sep 26, 2023
2dce538
Fix a couple of minor issues
wayfarer3130 Sep 27, 2023
94a0887
Merge remote-tracking branch 'ohif/master' into fix/stack-image-sync-for
wayfarer3130 Sep 27, 2023
7666b93
Fix a couple of dependencies on newer CS3D modules
wayfarer3130 Sep 27, 2023
32ddcc2
Remove changes other than stack sync
wayfarer3130 Sep 27, 2023
c42aefa
Minor spelling issues
wayfarer3130 Sep 27, 2023
f4a5f01
Removing a couple more unused changes
wayfarer3130 Sep 27, 2023
6a0470a
Merge branch 'master' of github.com:OHIF/Viewers into pr/fix/stack-im…
sedghi Sep 29, 2023
2e8e899
Merge branch 'master' into fix/stack-image-sync-for
wayfarer3130 Sep 29, 2023
480a147
Merge branch 'fix/stack-image-sync-for' of github.com:OHIF/Viewers in…
sedghi Sep 29, 2023
15cc74e
Added a bit of documentation
wayfarer3130 Sep 29, 2023
29f799c
upgrade cs3d
sedghi Oct 3, 2023
1948368
Merge branch 'fix/stack-image-sync-for' of github.com:OHIF/Viewers in…
sedghi Oct 3, 2023
46dc64c
Merge branch 'master' of github.com:OHIF/Viewers into pr/fix/stack-im…
sedghi Oct 3, 2023
7ff1bba
putting back stack prefetch listener for now
sedghi Oct 3, 2023
78caf6b
fix displayset message bug
sedghi Oct 3, 2023
c88dc96
Merge branch 'master' of github.com:OHIF/Viewers into pr/fix/stack-im…
sedghi Oct 3, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
51 changes: 48 additions & 3 deletions extensions/cornerstone/src/commandsModule.ts
wayfarer3130 marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -267,6 +267,18 @@ function commandsModule({
toolbarService.recordInteraction(props);
},

// Enable or disable a toggleable command, without calling the activation
// Used to setup already active tools from hanging protocols
toolbarSetActive: props => {
toolbarService.setActive(props.toolId, props.isActive ?? true);
},
wayfarer3130 marked this conversation as resolved.
Show resolved Hide resolved

// Enable stack prefetch on the given element
stackPrefetch: props => {
const { element } = props;
cstUtils.stackPrefetch.enable(element);
},

setToolActive: ({ toolName, toolGroupId = null, toggledState }) => {
if (toolName === 'Crosshairs') {
const activeViewportToolGroup = toolGroupService.getToolGroup(null);
Expand Down Expand Up @@ -345,6 +357,28 @@ function commandsModule({
],
});
},
toggleReferenceLines: ({ toggledState }) => {
try {
const { viewportId } = _getActiveViewportEnabledElement();
const toolGroup = toolGroupService.getToolGroupForViewport(viewportId);

if (!toggledState) {
toolGroup.setToolDisabled(ReferenceLinesTool.toolName);
return;
}

toolGroup.setToolConfiguration(
ReferenceLinesTool.toolName,
{
sourceViewportId: viewportId,
},
true // overwrite
);
toolGroup.setToolEnabled(ReferenceLinesTool.toolName);
} catch (e) {
console.log("Couldn't activate reference lines");
}
},
showDownloadViewportModal: () => {
const { activeViewportId } = viewportGridService.getState();

Expand Down Expand Up @@ -551,7 +585,6 @@ function commandsModule({

toggleStackImageSync: ({ toggledState }) => {
toggleStackImageSync({
getEnabledElement,
servicesManager,
toggledState,
});
Expand All @@ -563,6 +596,11 @@ function commandsModule({
const viewportId = viewportInfo.getViewportId();
const toolGroup = toolGroupService.getToolGroupForViewport(viewportId);

if (!toggledState) {
toolGroup.setToolDisabled(ReferenceLinesTool.toolName);
return;
}

toolGroup.setToolConfiguration(
ReferenceLinesTool.toolName,
{
Expand Down Expand Up @@ -701,8 +739,15 @@ function commandsModule({
},
storePresentation: {
commandFn: actions.storePresentation,
storeContexts: [],
options: {},
},
stackPrefetch: {
commandFn: actions.stackPrefetch,
},
toolbarSetActive: {
commandFn: actions.toolbarSetActive,
},
toggleReferenceLines: {
commandFn: actions.toggleReferenceLines,
},
};

Expand Down
69 changes: 40 additions & 29 deletions extensions/cornerstone/src/init.tsx
wayfarer3130 marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ import {
Settings,
utilities as csUtilities,
} from '@cornerstonejs/core';
import { Enums, utilities, ReferenceLinesTool } from '@cornerstonejs/tools';
import { Enums, } from '@cornerstonejs/tools';
import { cornerstoneStreamingImageVolumeLoader } from '@cornerstonejs/streaming-image-volume-loader';

import initWADOImageLoader from './initWADOImageLoader';
Expand Down Expand Up @@ -72,6 +72,7 @@ export default async function init({
cornerstoneViewportService,
hangingProtocolService,
toolGroupService,
toolbarService,
viewportGridService,
stateSyncService,
} = servicesManager.services as CornerstoneServices;
Expand Down Expand Up @@ -182,9 +183,45 @@ export default async function init({
commandsManager,
});

/**
* When a viewport gets a new display set, this call will go through all the
* active tools in the toolbar, and call any commands registered in the
* toolbar service with a callback to re-enable on displaying the viewport.
*/
const newStackCallback = evt => {
const { element } = evt.detail;
utilities.stackPrefetch.enable(element);
const activeTools = toolbarService.getActiveTools();

activeTools.forEach(tool => {
const toolData = toolbarService.getButton(tool);
const commands = toolData?.listeners?.newStack;
console.log('Running', tool, commands);
commandsManager.run(commands, { element, evt });
});
};

const activeViewportIdChangedListener = evt => {
const { viewportId } = evt;
const toolGroup = toolGroupService.getToolGroupForViewport(viewportId);

const activeTools = toolbarService.getActiveTools();

activeTools.forEach(tool => {
if (!toolGroup?._toolInstances?.[tool]) {
return;
}

// check if tool is active on the new viewport
const toolEnabled = toolGroup._toolInstances[tool].mode === Enums.ToolModes.Enabled;

if (!toolEnabled) {
return;
}

const toolData = toolbarService.getButton(tool);
const commands = toolData?.listeners?.activeViewportIdChanged;
commandsManager.run(commands, { viewportId, evt });
});
};

const resetCrosshairs = evt => {
Expand Down Expand Up @@ -236,33 +273,7 @@ export default async function init({

viewportGridService.subscribe(
viewportGridService.EVENTS.ACTIVE_VIEWPORT_ID_CHANGED,
({ viewportId }) => {
const toolGroup = toolGroupService.getToolGroupForViewport(viewportId);

if (!toolGroup || !toolGroup._toolInstances?.['ReferenceLines']) {
return;
}

// check if reference lines are active
const referenceLinesEnabled =
toolGroup._toolInstances['ReferenceLines'].mode === Enums.ToolModes.Enabled;

if (!referenceLinesEnabled) {
return;
}

toolGroup.setToolConfiguration(
ReferenceLinesTool.toolName,
{
sourceViewportId: viewportId,
},
true // overwrite
);

// make sure to set it to enabled again since we want to recalculate
// the source-target lines
toolGroup.setToolEnabled(ReferenceLinesTool.toolName);
}
activeViewportIdChangedListener
);
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -142,12 +142,16 @@ const CornerstoneViewportDownloadForm = ({
const properties = viewport.getProperties();

downloadViewport.setStack([imageId]).then(() => {
downloadViewport.setProperties(properties);

const newWidth = Math.min(width || image.width, MAX_TEXTURE_SIZE);
const newHeight = Math.min(height || image.height, MAX_TEXTURE_SIZE);

resolve({ width: newWidth, height: newHeight });
try {
wayfarer3130 marked this conversation as resolved.
Show resolved Hide resolved
downloadViewport.setProperties(properties);
const newWidth = Math.min(width || image.width, MAX_TEXTURE_SIZE);
const newHeight = Math.min(height || image.height, MAX_TEXTURE_SIZE);

resolve({ width: newWidth, height: newHeight });
} catch (e) {
// Happens on clicking the cancel button
console.warn('Unable to set properties', e);
}
});
} else if (downloadViewport instanceof VolumeViewport) {
const actors = viewport.getActors();
Expand Down
131 changes: 61 additions & 70 deletions extensions/cornerstone/src/utils/stackSync/toggleStackImageSync.ts
Original file line number Diff line number Diff line change
@@ -1,90 +1,81 @@
import calculateViewportRegistrations from './calculateViewportRegistrations';

// [ {
// synchronizerId: string,
// viewports: [ { viewportId: string, renderingEngineId: string, index: number } , ...]
// ]}
let STACK_IMAGE_SYNC_GROUPS_INFO = [];

export default function toggleStackImageSync({ toggledState, servicesManager, getEnabledElement }) {
const disableSync = (syncName, servicesManager) => {
wayfarer3130 marked this conversation as resolved.
Show resolved Hide resolved
const { syncGroupService, viewportGridService, displaySetService, cornerstoneViewportService } =
wayfarer3130 marked this conversation as resolved.
Show resolved Hide resolved
servicesManager.services;
const viewports = getViewports(viewportGridService, displaySetService);
viewports.forEach(gridViewport => {
const { viewportId } = gridViewport.viewportOptions;
const viewport = cornerstoneViewportService.getCornerstoneViewport(viewportId);
if (!viewport) {
return;
}
syncGroupService.removeViewportFromSyncGroup(
viewport.id,
viewport.getRenderingEngine().id,
syncName
);
});
};

if (!toggledState) {
STACK_IMAGE_SYNC_GROUPS_INFO.forEach(syncGroupInfo => {
const { viewports, synchronizerId } = syncGroupInfo;

viewports.forEach(({ viewportId, renderingEngineId }) => {
syncGroupService.removeViewportFromSyncGroup(viewportId, renderingEngineId, synchronizerId);
});
});

return;
}

STACK_IMAGE_SYNC_GROUPS_INFO = [];

// create synchronization groups and add viewports
const { viewports } = viewportGridService.getState();
const getViewports = (viewportGridService, displaySetService) => {
wayfarer3130 marked this conversation as resolved.
Show resolved Hide resolved
let { viewports } = viewportGridService.getState();

viewports = [...viewports.values()];
// filter empty viewports
const viewportsArray = Array.from(viewports.values())
.filter(viewport => viewport.displaySetInstanceUIDs?.length)
// filter reconstructable viewports
.filter(viewport => {
const { displaySetInstanceUIDs } = viewport;
viewports = viewports.filter(
viewport => viewport.displaySetInstanceUIDs && viewport.displaySetInstanceUIDs.length
);

for (const displaySetInstanceUID of displaySetInstanceUIDs) {
const displaySet = displaySetService.getDisplaySetByUID(displaySetInstanceUID);
// filter reconstructable viewports
viewports = viewports.filter(viewport => {
const { displaySetInstanceUIDs } = viewport;

return !!displaySet?.isReconstructable;
}
});
for (const displaySetInstanceUID of displaySetInstanceUIDs) {
const displaySet = displaySetService.getDisplaySetByUID(displaySetInstanceUID);

const viewportsByOrientation = viewportsArray.reduce((acc, viewport) => {
const { viewportId, viewportType } = viewport.viewportOptions;

if (viewportType !== 'stack') {
console.warn('Viewport is not a stack, cannot sync images yet');
return acc;
}

const { element } = cornerstoneViewportService.getViewportInfo(viewportId);
const { viewport: csViewport, renderingEngineId } = getEnabledElement(element);
const { viewPlaneNormal } = csViewport.getCamera();

// Should we round here? I guess so, but not sure how much precision we need
const orientation = viewPlaneNormal.map(v => Math.round(v)).join(',');
// TODO - add a better test than isReconstructable
if (displaySet && displaySet.isReconstructable) {
return true;
}

if (!acc[orientation]) {
acc[orientation] = [];
return false;
}
});

acc[orientation].push({ viewportId, renderingEngineId });
// viewports = viewports.filter(viewport => viewport.viewportOptions.viewportType === 'stack');

return acc;
}, {});
console.log('viewports=', viewports);
return viewports;
};

// create synchronizer for each group
Object.values(viewportsByOrientation).map(viewports => {
let synchronizerId = viewports.map(({ viewportId }) => viewportId).join(',');
const STACK_SYNC_NAME = 'stackImageSync';

synchronizerId = `imageSync_${synchronizerId}`;
export default function toggleStackImageSync({
toggledState,
servicesManager,
viewports: providedViewports,
}) {
if (!toggledState) {
return disableSync(STACK_SYNC_NAME, servicesManager);
}

calculateViewportRegistrations(viewports);
console.log('Toggling stack image sync');
const { syncGroupService, viewportGridService, displaySetService, cornerstoneViewportService } =
servicesManager.services;

viewports.forEach(({ viewportId, renderingEngineId }) => {
syncGroupService.addViewportToSyncGroup(viewportId, renderingEngineId, {
type: 'stackimage',
id: synchronizerId,
source: true,
target: true,
});
});
const viewports = providedViewports || getViewports(viewportGridService, displaySetService);

STACK_IMAGE_SYNC_GROUPS_INFO.push({
synchronizerId,
viewports,
// create synchronization group and add the viewports to it.
viewports.forEach(gridViewport => {
const { viewportId } = gridViewport.viewportOptions;
const viewport = cornerstoneViewportService.getCornerstoneViewport(viewportId);
if (!viewport) {
return;
}
syncGroupService.addViewportToSyncGroup(viewportId, viewport.getRenderingEngine().id, {
type: 'stackimage',
id: STACK_SYNC_NAME,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I liked the pattern we had before

  let synchronizerId = viewports.map(({ viewportId }) => viewportId).join(',');

    synchronizerId = `imageSync_${synchronizerId}`;

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is a fixed name - there is a single synchronizer used for all viewports. The synchronizer itself decides when it is safe to run, which prevents duplication of code - we previously had places where there was different logic on both sides doing slightly different things, and depending on ordering you could end up with different results.

source: true,
target: true,
});
});
}
Loading