Skip to content

Commit

Permalink
Merge pull request #560 from PaulHax/layer-fix
Browse files Browse the repository at this point in the history
Fix adding DICOM dataset as layer
  • Loading branch information
floryst authored Mar 18, 2024
2 parents 3cadfb0 + b0a1b15 commit 72d2e79
Show file tree
Hide file tree
Showing 10 changed files with 134 additions and 23 deletions.
1 change: 1 addition & 0 deletions src/components/LayerProperties.vue
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,7 @@ export default defineComponent({
thumb-label
:model-value="blendConfig.opacity"
@update:model-value="setBlendConfig('opacity', $event)"
data-testid="layer-opacity-slider"
/>
</div>
</template>
6 changes: 5 additions & 1 deletion src/components/ModulePanel.vue
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,11 @@
icons-and-text
show-arrows
>
<v-tab v-for="item in Modules" :key="item.name">
<v-tab
v-for="item in Modules"
:key="item.name"
:data-testid="`module-tab-${item.name}`"
>
<div class="tab-content">
<span class="mb-0 mt-1 module-text">{{ item.name }}</span>
<v-icon>mdi-{{ item.icon }}</v-icon>
Expand Down
2 changes: 2 additions & 0 deletions src/components/PatientStudyVolumeBrowser.vue
Original file line number Diff line number Diff line change
Expand Up @@ -291,12 +291,14 @@ export default defineComponent({
size="x-small"
class="dataset-menu"
@click.stop
data-testid="dataset-menu-button"
>
<v-menu activator="parent">
<v-list>
<v-list-item
v-if="volume.layerable"
@click.stop="volume.layerHandler()"
data-testid="dataset-menu-layer-item"
>
<template v-if="volume.layerLoading">
<div style="margin: 0 auto">
Expand Down
10 changes: 8 additions & 2 deletions src/composables/useCurrentImage.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,11 @@ import {
} from '@/src/store/datasets-images';
import { useLayersStore } from '@/src/store/datasets-layers';
import { createLPSBounds, getAxisBounds } from '@/src/utils/lps';
import { getImageID, useDatasetStore } from '@/src/store/datasets';
import {
getDataSelection,
getImageID,
useDatasetStore,
} from '@/src/store/datasets';
import { storeToRefs } from 'pinia';

export interface CurrentImageContext {
Expand Down Expand Up @@ -66,9 +70,11 @@ export function getIsImageLoading(imageID: Maybe<string>) {
}

export function getImageLayers(imageID: Maybe<string>) {
if (!imageID) return [];
const selection = getDataSelection(imageID);
const layersStore = useLayersStore();
return layersStore
.getLayers(imageID ? { type: 'image', dataID: imageID } : null)
.getLayers(selection)
.filter(({ id }) => id in layersStore.layerImages);
}

Expand Down
2 changes: 1 addition & 1 deletion src/io/manifest.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { z } from 'zod';

const RemoteResource = z.object({
export const RemoteResource = z.object({
url: z.string(),
name: z.optional(z.string()),
});
Expand Down
9 changes: 9 additions & 0 deletions src/store/datasets.ts
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,15 @@ export const getDataID = (imageID: string) => {
return dicomStore.imageIDToVolumeKey[imageID] ?? imageID;
};

export const getDataSelection = (imageID: string) => {
const dicomStore = useDICOMStore();
const volumeKey = dicomStore.imageIDToVolumeKey[imageID];
if (volumeKey) {
return makeDICOMSelection(volumeKey);
}
return makeImageSelection(imageID);
};

// Pass VolumeKey or ImageID and get ImageID
export const findImageID = (dataID: string) => {
const dicomStore = useDICOMStore();
Expand Down
12 changes: 12 additions & 0 deletions tests/pageobjects/volview.page.ts
Original file line number Diff line number Diff line change
Expand Up @@ -149,6 +149,18 @@ class VolViewPage extends Page {
get editLabelModalDoneButton() {
return $('button[data-testid="edit-label-done-button"]');
}

get datasetMenuButtons() {
return $$('button[data-testid="dataset-menu-button"]');
}

get renderingModuleTab() {
return $('button[data-testid="module-tab-Rendering"]');
}

get layerOpacitySliders() {
return $$('div[data-testid="layer-opacity-slider"] input');
}
}

export const volViewPage = new VolViewPage();
Expand Down
60 changes: 60 additions & 0 deletions tests/specs/layers.e2e.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
import { volViewPage } from '../pageobjects/volview.page';
import { openUrls } from './utils';

describe('Add Layer button', () => {
it('should create overlay with 2 DICOM images', async () => {
await openUrls([
{
url: 'https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download',
name: 'prostate.zip',
},
{
url: 'https://data.kitware.com/api/v1/item/6352a2b311dab8142820a33b/download',
name: 'MRA-Head_and_Neck.zip',
},
]);

// Wait for both volumes to appear in list
await browser.waitUntil(
async () => {
const menus = await volViewPage.datasetMenuButtons;
return menus.length >= 2;
},
{
timeout: 30000,
timeoutMsg: `Expected 2 volumes to appear in list`,
}
);

const menus = await volViewPage.datasetMenuButtons;
await menus[0].click();
await browser.waitUntil(
async () => {
const addLayerButton = await $(
'div[data-testid="dataset-menu-layer-item"]'
);
return addLayerButton.isClickable();
},
{ timeoutMsg: 'Expected clickable Add Layer button' }
);

const addLayerButton = await $(
'div[data-testid="dataset-menu-layer-item"]'
);
await addLayerButton.click();

const renderTab = await volViewPage.renderingModuleTab;
await renderTab.click();

// need to wait a little for layer section to render
await browser.waitUntil(
async function slidersExist() {
const layerOpacitySliders = await volViewPage.layerOpacitySliders;
return layerOpacitySliders.length > 0;
},
{
timeoutMsg: `Expected at least one layer opacity slider`,
}
);
});
});
20 changes: 1 addition & 19 deletions tests/specs/remote-manifest.e2e.ts
Original file line number Diff line number Diff line change
@@ -1,23 +1,5 @@
import * as path from 'path';
import * as fs from 'fs';
import { cleanuptotal } from 'wdio-cleanuptotal-service';
import { TEMP_DIR } from '../../wdio.shared.conf';
import { volViewPage } from '../pageobjects/volview.page';
import { downloadFile } from './utils';

async function writeManifestToFile(manifest: any, fileName: string) {
const filePath = path.join(TEMP_DIR, fileName);
await fs.promises.writeFile(filePath, JSON.stringify(manifest));
cleanuptotal.addCleanup(async () => {
fs.unlinkSync(filePath);
});
return filePath;
}

async function openVolViewPage(fileName: string) {
const urlParams = `?urls=[tmp/${fileName}]`;
await volViewPage.open(urlParams);
}
import { downloadFile, writeManifestToFile, openVolViewPage } from './utils';

describe('VolView loading of remoteManifest.json', () => {
it('should show error when there is no name and URL is malformed', async () => {
Expand Down
35 changes: 35 additions & 0 deletions tests/specs/utils.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,10 @@
import * as path from 'path';
import * as fs from 'fs';
import { z } from 'zod';
import { cleanuptotal } from 'wdio-cleanuptotal-service';
import { TEMP_DIR } from '../../wdio.shared.conf';
import { volViewPage } from '../pageobjects/volview.page';
import { RemoteResource } from '../../src/io/manifest';

// File is not automatically deleted
export const downloadFile = async (url: string, fileName: string) => {
Expand All @@ -13,3 +17,34 @@ export const downloadFile = async (url: string, fileName: string) => {
}
return savePath;
};

export async function writeManifestToFile(manifest: any, fileName: string) {
const filePath = path.join(TEMP_DIR, fileName);
await fs.promises.writeFile(filePath, JSON.stringify(manifest));
cleanuptotal.addCleanup(async () => {
fs.unlinkSync(filePath);
});
return filePath;
}

export async function openVolViewPage(fileName: string) {
const urlParams = `?urls=[tmp/${fileName}]`;
await volViewPage.open(urlParams);
}

type RemoteResourceType = z.infer<typeof RemoteResource> & { name: string };

export async function openUrls(urlsAndNames: Array<RemoteResourceType>) {
await Promise.all(
urlsAndNames.map((resource) => downloadFile(resource.url, resource.name))
);

const resources = urlsAndNames.map(({ name }) => ({ url: `/tmp/${name}` }));
const manifest = {
resources,
};
const fileName = 'openUrlsManifest.json';
await writeManifestToFile(manifest, fileName);
await openVolViewPage(fileName);
await volViewPage.waitForViews();
}

0 comments on commit 72d2e79

Please sign in to comment.