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

feat: Add simple multi-monitor support #4178

Open
wants to merge 35 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 18 commits
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
ed09808
feat: Multi monitor launch
wayfarer3130 May 27, 2024
efe4abd
fix: Added update to existing window
wayfarer3130 May 27, 2024
b6cf4c0
feat: Add preserved multimonitor mode
wayfarer3130 May 27, 2024
26d181d
Fix the default mode to allow multimonitor
wayfarer3130 May 27, 2024
06a4fae
Merge remote-tracking branch 'origin/master' into feat/multi-monitor-…
wayfarer3130 May 27, 2024
9abbc6c
Fix load with multimonitor not configured
wayfarer3130 May 27, 2024
bcfed7e
Exclude the app-config from terser mucking so it is readable.
wayfarer3130 May 27, 2024
ab789d7
fix tests
wayfarer3130 May 27, 2024
15d5744
Undid accidental removal of name in package.json
wayfarer3130 May 27, 2024
7052f31
Remove console logs
wayfarer3130 May 28, 2024
00419a4
Merge remote-tracking branch 'ohif/master' into feat/multi-monitor-take2
wayfarer3130 Aug 6, 2024
e08ea76
docs
wayfarer3130 Nov 7, 2024
3bc9025
Merge remote-tracking branch 'origin/master' into feat/multi-monitor-…
wayfarer3130 Nov 25, 2024
709c7c1
fix: Launch using new multi-monitor setup
wayfarer3130 Nov 26, 2024
8eb5950
Merge remote-tracking branch 'origin/master' into feat/multi-monitor-…
wayfarer3130 Nov 29, 2024
d18ae3a
Merge remote-tracking branch 'origin/master' into feat/multi-monitor-…
wayfarer3130 Dec 4, 2024
6f7e5bf
fix: Allow commandsManager commands to run on other window
wayfarer3130 Dec 4, 2024
652b3e6
fix: Preserve annotations by allowing commands to be run in other window
wayfarer3130 Dec 4, 2024
d3b0df6
Merge remote-tracking branch 'origin/master' into feat/multi-monitor-…
wayfarer3130 Dec 12, 2024
d7aec61
fix: Preserve multimonitor configuration
wayfarer3130 Dec 12, 2024
fd17be7
Fix paths when publicUrl is set
wayfarer3130 Dec 12, 2024
90522c1
Merge remote-tracking branch 'origin/master' into feat/multi-monitor-…
wayfarer3130 Dec 13, 2024
46ec727
fix: Other monitor show command
wayfarer3130 Dec 13, 2024
96bae40
Remove console log
wayfarer3130 Dec 19, 2024
ebddf4d
Merge remote-tracking branch 'origin/master' into feat/multi-monitor-…
wayfarer3130 Dec 19, 2024
8a680dc
fix: Multi-monitor - mostly working variant
wayfarer3130 Dec 20, 2024
513d8c0
fix: Add some default options
wayfarer3130 Dec 20, 2024
351c909
Relaunch without refresh in any windows
wayfarer3130 Dec 20, 2024
7d06e7b
fix: Apply full hanging protocol when changing studies
wayfarer3130 Dec 30, 2024
879b221
Merge remote-tracking branch 'origin/master' into feat/multi-monitor-…
wayfarer3130 Dec 30, 2024
ddc7787
fix: loadStudy wasn't working on same window
wayfarer3130 Dec 31, 2024
f5e8873
Add the simplest multi-monitor display
wayfarer3130 Dec 31, 2024
cfa1a6a
fix: Improve context menu behavior by checking for annotation lock st…
sedghi Jan 9, 2025
9d09315
Merge branch 'master' of github.com:OHIF/Viewers into feat/multi-moni…
sedghi Jan 9, 2025
cf6640d
Merge remote-tracking branch 'origin/master' into feat/multi-monitor-…
wayfarer3130 Jan 10, 2025
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
1 change: 1 addition & 0 deletions .webpack/webpack.base.js
Original file line number Diff line number Diff line change
Expand Up @@ -218,6 +218,7 @@ module.exports = (env, argv, { SRC_DIR, ENTRY }) => {
config.optimization.minimizer = [
new TerserJSPlugin({
parallel: true,
exclude: /app-config.js/,
Copy link
Member

Choose a reason for hiding this comment

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

I guess this is only in debug mode? Sometimes the app config is huge and it is desirable to minify it

terserOptions: {},
}),
];
Expand Down
4 changes: 2 additions & 2 deletions extensions/default/src/Panels/WrappedPanelStudyBrowser.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -13,10 +13,10 @@ import requestDisplaySetCreationForStudy from './requestDisplaySetCreationForStu
* @param {object} commandsManager
* @param {object} extensionManager
*/
function WrappedPanelStudyBrowser({ commandsManager, extensionManager, servicesManager }) {
function WrappedPanelStudyBrowser({ extensionManager, servicesManager }) {
// TODO: This should be made available a different way; route should have
// already determined our datasource
const dataSource = extensionManager.getDataSources()[0];
const [dataSource] = extensionManager.getActiveDataSource();
const _getStudiesForPatientByMRN = getStudiesForPatientByMRN.bind(null, dataSource);
const _getImageSrcFromImageId = useCallback(
_createGetImageSrcFromImageIdFn(extensionManager),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ function requestDisplaySetCreationForStudy(
return;
}

dataSource.retrieve.series.metadata({ StudyInstanceUID, madeInClient });
return dataSource.retrieve.series.metadata({ StudyInstanceUID, madeInClient });
Copy link
Member

Choose a reason for hiding this comment

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

There seems to be two files named requestDisplaySetCreationForStudy , can we re-use one

}

export default requestDisplaySetCreationForStudy;
2 changes: 1 addition & 1 deletion extensions/default/src/ViewerLayout/ViewerHeader.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ function ViewerHeader({
if (dataSourceIdx !== -1 && existingDataSource) {
searchQuery.append('datasources', pathname.substring(dataSourceIdx + 1));
}

servicesManager.services.multiMonitorService.appendQuery(searchQuery);
if (configUrl) {
searchQuery.append('configUrl', configUrl);
}
Expand Down
60 changes: 59 additions & 1 deletion extensions/default/src/commandsModule.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { Types } from '@ohif/core';
import { Types, DicomMetadataStore } from '@ohif/core';

import { ContextMenuController, defaultContextMenu } from './CustomizableContextMenu';
import DicomTagBrowser from './DicomTagBrowser/DicomTagBrowser';
Expand All @@ -16,6 +16,7 @@ import { useHangingProtocolStageIndexStore } from './stores/useHangingProtocolSt
import { useToggleHangingProtocolStore } from './stores/useToggleHangingProtocolStore';
import { useViewportsByPositionStore } from './stores/useViewportsByPositionStore';
import { useToggleOneUpViewportGridStore } from './stores/useToggleOneUpViewportGridStore';
import requestDisplaySetCreationForStudy from './Panels/requestDisplaySetCreationForStudy';

export type HangingProtocolParams = {
protocolId?: string;
Expand All @@ -25,6 +26,26 @@ export type HangingProtocolParams = {
reset?: false;
};

/**
* The studies from display sets gets the studies in study date
* order or in study instance UID order - not very useful, but
* if not specifically specified then at least making it consistent is useful.
*/
const getStudiesfromDisplaySets = displaySets => {
const studyMap = {};

const ret = displaySets.reduce((prev, curr) => {
const { StudyInstanceUID } = curr;
if (!studyMap[StudyInstanceUID]) {
const study = DicomMetadataStore.getStudy(StudyInstanceUID);
studyMap[StudyInstanceUID] = study;
prev.push(study);
}
return prev;
}, []);
return ret;
};

export type UpdateViewportDisplaySetParams = {
direction: number;
excludeNonImageModalities?: boolean;
Expand All @@ -33,6 +54,7 @@ export type UpdateViewportDisplaySetParams = {
const commandsModule = ({
servicesManager,
commandsManager,
extensionManager,
}: Types.Extensions.ExtensionParams): Types.Extensions.CommandsModule => {
const {
customizationService,
Expand All @@ -41,12 +63,42 @@ const commandsModule = ({
uiNotificationService,
viewportGridService,
displaySetService,
multiMonitorService,
} = servicesManager.services;

// Define a context menu controller for use with any context menus
const contextMenuController = new ContextMenuController(servicesManager, commandsManager);

const actions = {
/**
* Runs a command in multi-monitor mode. No-op if not multi-monitor.
*/
multimonitor: async options => {
const { commands, screenDelta, studyInstanceUID } = options;
if (multiMonitorService.numberOfScreens < 2) {
return options.fallback?.(options);
}

await multiMonitorService.launchWindow(studyInstanceUID, screenDelta, options);
if (commands) {
multiMonitorService.run(screenDelta, commands, options);
}
},

/** Ensures that the specified study is available for display */
loadStudy: async options => {
const { studyInstanceUID } = options;
if (hangingProtocolService.hasStudyUID(studyInstanceUID)) {
return;
}
const [dataSource] = extensionManager.getActiveDataSource();
await requestDisplaySetCreationForStudy(dataSource, displaySetService, studyInstanceUID);
const activeStudy = DicomMetadataStore.getStudy(studyInstanceUID);
hangingProtocolService.addStudy(activeStudy);
const displaySets = displaySetService.getActiveDisplaySets();
hangingProtocolService.setDisplaySets(displaySets);
},

/**
* Show the context menu.
* @param options.menuId defines the menu name to lookup, from customizationService
Expand Down Expand Up @@ -562,6 +614,12 @@ const commandsModule = ({
};

const definitions = {
multimonitor: {
commandFn: actions.multimonitor,
},
loadStudy: {
commandFn: actions.loadStudy,
},
showContextMenu: {
commandFn: actions.showContextMenu,
},
Expand Down
16 changes: 4 additions & 12 deletions extensions/default/src/hangingprotocols/hpMNGrid.ts
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,7 @@ const hpMN: Types.HangingProtocol.Protocol = {
},
stages: [
{
id: '2x2',
name: '2x2',
name: '2x2',
stageActivation: {
enabled: {
Expand Down Expand Up @@ -174,11 +174,7 @@ const hpMN: Types.HangingProtocol.Protocol = {

// 3x1 stage
{
id: '3x1',
// Obsolete settings:
requiredViewports: 1,
preferredViewports: 3,
// New equivalent:
name: '3x1',
stageActivation: {
enabled: {
minViewportsMatched: 3,
Expand Down Expand Up @@ -232,9 +228,7 @@ const hpMN: Types.HangingProtocol.Protocol = {

// A 2x1 stage
{
id: '2x1',
requiredViewports: 1,
preferredViewports: 2,
name: '2x1',
stageActivation: {
enabled: {
minViewportsMatched: 2,
Expand Down Expand Up @@ -276,9 +270,7 @@ const hpMN: Types.HangingProtocol.Protocol = {

// A 1x1 stage - should be automatically activated if there is only 1 viewable instance
{
id: '1x1',
requiredViewports: 1,
preferredViewports: 1,
name: '1x1',
stageActivation: {
enabled: {
minViewportsMatched: 1,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@ import { StudyBrowser } from '@ohif/ui-next';
import { useTrackedMeasurements } from '../../getContextModule';
import { Separator } from '@ohif/ui-next';
import { PanelStudyBrowserHeader } from '@ohif/extension-default';
import { useAppConfig } from '@state';
import { defaultActionIcons, defaultViewPresets } from './constants';

const { formatDate, createStudyBrowserTabs } = utils;
Expand Down Expand Up @@ -389,6 +388,29 @@ export default function PanelStudyBrowserTracking({
}
}

const _launchMultiMonitor = (studyInstanceUID, screenDelta) => {
commandsManager.run('multimonitor', {
studyInstanceUID,
screenDelta,
activeStudyUID: studyInstanceUID,
fallback: () => _handleStudyClick(studyInstanceUID),
commands: [
{
commandName: 'loadStudy',
commandOptions: {
protocolId: '@ohif/mnGrid',
},
},
{
commandName: 'setHangingProtocol',
commandOptions: {
protocolId: '@ohif/mnGrid',
},
},
],
});
};

useEffect(() => {
if (jumpToDisplaySet) {
// Get element by displaySetInstanceUID
Expand Down Expand Up @@ -511,6 +533,7 @@ export default function PanelStudyBrowserTracking({
activeTabName={activeTabName}
expandedStudyInstanceUIDs={expandedStudyInstanceUIDs}
onClickStudy={_handleStudyClick}
onClickStudyInfo={_launchMultiMonitor}
onClickTab={clickedTabName => {
setActiveTabName(clickedTabName);
}}
Expand Down
65 changes: 65 additions & 0 deletions platform/app/public/config/default.js
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,71 @@ window.config = {
prefetch: 25,
},
// filterQueryParam: false,
// Defines multi-monitor layouts
multimonitor: [
{
id: 'split',
test: ({ multimonitor }) => multimonitor === 'split',
screens: [
{
id: 'primary',
// This is the primary screen, so don't launch is separately, but use primary
launch: 'primary',
screen: null,
location: {
screen: 0,
width: 0.5,
height: 1,
left: 0,
top: 0,
},
},
{
id: 'secondary',
// This is a window instance, so launch as a url
launch: 'url',
screen: null,
location: {
width: 0.48,
height: 1,
left: 0.52,
top: 0,
},
},
],
},

{
id: '2',
test: ({ multimonitor }) => multimonitor === '2',
screens: [
{
id: 'primary',
// This is the primary screen, so don't launch is separately, but use primary
launch: 'primary',
screen: 0,
location: {
width: 100,
height: 100,
left: 0,
top: 0,
},
},
{
id: 'secondary',
// This is a window instance, so launch as a url
launch: 'url',
screen: 1,
location: {
width: 100,
height: 100,
left: 0,
top: 0,
},
},
],
},
],
defaultDataSourceName: 'dicomweb',
/* Dynamic config allows user to pass "configUrl" query string this allows to load config without recompiling application. The regex will ensure valid configuration source */
// dangerouslyUseDynamicConfig: {
Expand Down
65 changes: 65 additions & 0 deletions platform/app/public/config/e2e.js
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,71 @@ window.config = {
investigationalUseDialog: {
option: 'never',
},
// Defines multi-monitor layouts
multimonitor: [
{
id: 'split',
test: ({ multimonitor }) => multimonitor === 'split',
screens: [
{
id: 'primary',
// This is the primary screen, so don't launch is separately, but use primary
launch: 'primary',
screen: null,
location: {
screen: 0,
width: 0.5,
height: 1,
left: 0,
top: 0,
},
},
{
id: 'secondary',
// This is a window instance, so launch as a url
launch: 'url',
screen: null,
location: {
width: 0.48,
height: 1,
left: 0.52,
top: 0,
},
},
],
},

{
id: '2',
test: ({ multimonitor }) => multimonitor === '2',
screens: [
{
id: 'primary',
// This is the primary screen, so don't launch is separately, but use primary
launch: 'primary',
screen: 0,
location: {
width: 100,
height: 100,
left: 0,
top: 0,
},
},
{
id: 'secondary',
// This is a window instance, so launch as a url
launch: 'url',
screen: 1,
location: {
width: 100,
height: 100,
left: 0,
top: 0,
},
},
],
},
],
dataSources: [
{
namespace: '@ohif/extension-default.dataSourcesModule.dicomweb',
Expand Down
2 changes: 2 additions & 0 deletions platform/app/src/appInit.js
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ import {
PanelService,
WorkflowStepsService,
StudyPrefetcherService,
MultiMonitorService,
// utils,
} from '@ohif/core';

Expand Down Expand Up @@ -58,6 +59,7 @@ async function appInit(appConfigOrFunc, defaultExtensions, defaultModes) {
servicesManager.setExtensionManager(extensionManager);

servicesManager.registerServices([
[MultiMonitorService.REGISTRATION, appConfig.multimonitor],
UINotificationService.REGISTRATION,
UIModalService.REGISTRATION,
UIDialogService.REGISTRATION,
Expand Down
Loading
Loading