From 0ac45b0a4ed8299cf900fe8997941738dad8d52d Mon Sep 17 00:00:00 2001 From: Paul Elliott Date: Wed, 13 Nov 2024 11:11:50 -0500 Subject: [PATCH] feat(windowing): add forced window/level to JSON config --- .../useWindowingConfigInitializer.ts | 8 ++++++++ src/io/import/configJson.ts | 19 +++++++++++++++++++ src/store/view-configs/windowing.ts | 10 +++++++++- 3 files changed, 36 insertions(+), 1 deletion(-) diff --git a/src/composables/useWindowingConfigInitializer.ts b/src/composables/useWindowingConfigInitializer.ts index 0208fc0b6..ce534574e 100644 --- a/src/composables/useWindowingConfigInitializer.ts +++ b/src/composables/useWindowingConfigInitializer.ts @@ -124,6 +124,14 @@ export function useWindowingConfigInitializer( }, }); } + const forcedWL = store.runtimeConfigWindowLevel; + if (forcedWL) { + store.updateConfig(viewIdVal, imageIdVal, { + preset: { + ...forcedWL, + }, + }); + } store.resetWindowLevel(viewIdVal, imageIdVal); }); diff --git a/src/io/import/configJson.ts b/src/io/import/configJson.ts index 9ac10ee94..c23681a40 100644 --- a/src/io/import/configJson.ts +++ b/src/io/import/configJson.ts @@ -9,6 +9,7 @@ import { useRulerStore } from '@/src/store/tools/rulers'; import { useDataBrowserStore } from '@/src/store/data-browser'; import { usePolygonStore } from '@/src/store/tools/polygons'; import { useViewStore } from '@/src/store/views'; +import { useWindowingStore } from '@/src/store/view-configs/windowing'; import { actionToKey } from '@/src/composables/useKeyboardShortcuts'; import { useSegmentGroupStore } from '@/src/store/segmentGroups'; import { AnnotationToolStore } from '@/src/store/tools/useAnnotationTool'; @@ -70,12 +71,23 @@ const io = z }) .optional(); +// -------------------------------------------------------------------------- +// Window Level + +const windowing = z + .object({ + level: z.number(), + width: z.number(), + }) + .optional(); + export const config = z.object({ layout, dataBrowser, labels, shortcuts, io, + windowing, }); export type Config = z.infer; @@ -140,10 +152,17 @@ const applyIo = (manifest: Config) => { useLoadDataStore().segmentGroupExtension = manifest.io.segmentGroupExtension; }; +const applyWindowing = (manifest: Config) => { + if (!manifest.windowing) return; + + useWindowingStore().runtimeConfigWindowLevel = manifest.windowing; +}; + export const applyConfig = (manifest: Config) => { applyLayout(manifest); applyLabels(manifest); applySampleData(manifest); applyShortcuts(manifest); applyIo(manifest); + applyWindowing(manifest); }; diff --git a/src/store/view-configs/windowing.ts b/src/store/view-configs/windowing.ts index 6b38617b8..935c3fa32 100644 --- a/src/store/view-configs/windowing.ts +++ b/src/store/view-configs/windowing.ts @@ -24,9 +24,15 @@ export const defaultWindowLevelConfig = (): WindowLevelConfig => ({ }, }); -const useWindowingStore = defineStore('windowing', () => { +type WindowLevel = { + width: number; + level: number; +}; + +export const useWindowingStore = defineStore('windowing', () => { const configs = reactive>({}); const syncAcrossViews = ref(true); + const runtimeConfigWindowLevel = ref(); const setSyncAcrossViews = (yn: boolean) => { syncAcrossViews.value = yn; @@ -69,6 +75,7 @@ const useWindowingStore = defineStore('windowing', () => { } }; + // not really reset, actually translate config object into W/L const resetWindowLevel = (viewID: string, dataID: string) => { const config = configs[viewID]?.[dataID]; if (config == null) return; @@ -105,6 +112,7 @@ const useWindowingStore = defineStore('windowing', () => { }; return { + runtimeConfigWindowLevel, configs, getConfig, setSyncAcrossViews,