diff --git a/src/components/AnnotationCategoryItem.tsx b/src/components/AnnotationCategoryItem.tsx index a5ef931..2adadee 100644 --- a/src/components/AnnotationCategoryItem.tsx +++ b/src/components/AnnotationCategoryItem.tsx @@ -18,6 +18,7 @@ const AnnotationCategoryItem = ({ [annotationUID: string]: { opacity: number color: number[] + contourOnly: boolean } } checkedAnnotationUids: Set diff --git a/src/components/AnnotationCategoryList.tsx b/src/components/AnnotationCategoryList.tsx index f062f6f..4f6c823 100644 --- a/src/components/AnnotationCategoryList.tsx +++ b/src/components/AnnotationCategoryList.tsx @@ -80,6 +80,7 @@ const AnnotationCategoryList = ({ [annotationUID: string]: { opacity: number color: number[] + contourOnly: boolean } } checkedAnnotationUids: Set diff --git a/src/components/ColorSettingsMenu.tsx b/src/components/ColorSettingsMenu.tsx index 09e986c..d772a2e 100644 --- a/src/components/ColorSettingsMenu.tsx +++ b/src/components/ColorSettingsMenu.tsx @@ -1,11 +1,12 @@ import React from 'react' -import { Col, Divider, InputNumber, Row, Slider } from 'antd' +import { Checkbox, Col, Divider, InputNumber, Row, Slider } from 'antd' interface ColorSettingsMenuProps { annotationGroupsUIDs: string[] defaultStyle: { opacity: number color: number[] + contourOnly: boolean } onStyleChange: Function } @@ -14,6 +15,7 @@ interface ColorSettingsMenuState { currentStyle: { opacity: number color?: number[] + contourOnly: boolean } } @@ -34,7 +36,8 @@ ColorSettingsMenuState this.state = { currentStyle: { opacity: this.props.defaultStyle.opacity, - color: this.props.defaultStyle.color + color: this.props.defaultStyle.color, + contourOnly: this.props.defaultStyle.contourOnly } } } @@ -46,16 +49,12 @@ ColorSettingsMenuState uid, styleOptions: { color: this.state.currentStyle.color, - opacity: value + opacity: value, + contourOnly: this.state.currentStyle.contourOnly } }) }) - this.setState({ - currentStyle: { - opacity: value, - color: this.state.currentStyle.color - } - }) + this.updateCurrentStyle({ opacity: value }) } } @@ -66,18 +65,14 @@ ColorSettingsMenuState this.state.currentStyle.color[1], this.state.currentStyle.color[2] ] - this.setState((state) => ({ - currentStyle: { - color: color, - opacity: state.currentStyle.opacity - } - })) + this.updateCurrentStyle({ color }) this.props.annotationGroupsUIDs.forEach((uid) => { this.props.onStyleChange({ uid, styleOptions: { color: color, - opacity: this.state.currentStyle.opacity + opacity: this.state.currentStyle.opacity, + contourOnly: this.state.currentStyle.contourOnly } }) }) @@ -91,18 +86,14 @@ ColorSettingsMenuState Array.isArray(value) ? value[0] : value, this.state.currentStyle.color[2] ] - this.setState((state) => ({ - currentStyle: { - color: color, - opacity: state.currentStyle.opacity - } - })) + this.updateCurrentStyle({ color }) this.props.annotationGroupsUIDs.forEach((uid) => { this.props.onStyleChange({ uid, styleOptions: { color: color, - opacity: this.state.currentStyle.opacity + opacity: this.state.currentStyle.opacity, + contourOnly: this.state.currentStyle.contourOnly } }) }) @@ -116,25 +107,35 @@ ColorSettingsMenuState this.state.currentStyle.color[1], Array.isArray(value) ? value[0] : value ] - this.setState((state) => ({ - currentStyle: { - color: color, - opacity: state.currentStyle.opacity - } - })) - + this.updateCurrentStyle({ color }) this.props.annotationGroupsUIDs.forEach((uid) => { this.props.onStyleChange({ uid, styleOptions: { color: color, - opacity: this.state.currentStyle.opacity + opacity: this.state.currentStyle.opacity, + contourOnly: this.state.currentStyle.contourOnly } }) }) } } + handleShowOutlineOnly (value: boolean): void { + this.updateCurrentStyle({ contourOnly: value }) + + this.props.annotationGroupsUIDs.forEach((uid) => { + this.props.onStyleChange({ + uid, + styleOptions: { + color: this.state.currentStyle.color, + opacity: this.state.currentStyle.opacity, + contourOnly: value + } + }) + }) + } + getCurrentColor (): string { const rgb2hex = (values: number[]): string => { const r = values[0] @@ -150,6 +151,24 @@ ColorSettingsMenuState } } + updateCurrentStyle ({ + color, + opacity, + contourOnly + }: { + color?: number[] + opacity?: number + contourOnly?: boolean + }): void { + this.setState((state) => ({ + currentStyle: { + opacity: opacity ?? state.currentStyle.opacity, + color: color ?? state.currentStyle.color, + contourOnly: contourOnly ?? state.currentStyle.contourOnly + } + })) + } + render (): React.ReactNode { let colorSettings if (this.state.currentStyle.color != null) { @@ -259,6 +278,15 @@ ColorSettingsMenuState /> + + + this.handleShowOutlineOnly(event.target.checked)} + > + Show outline only + + ) } diff --git a/src/components/HoveredRoiTooltip.tsx b/src/components/HoveredRoiTooltip.tsx index 7ef12ef..4799568 100644 --- a/src/components/HoveredRoiTooltip.tsx +++ b/src/components/HoveredRoiTooltip.tsx @@ -1,11 +1,11 @@ const HoveredRoiTooltip = ({ xPosition, yPosition, - attributes + rois }: { xPosition: number yPosition: number - attributes: Array<{ name: string, value: string }> + rois: Array<{ index: number, roiUid: string, attributes: Array<{ name: string, value: string }>}> }): JSX.Element => { return (
- {attributes.map((attr) => ( -
- {attr.name}: {attr.value} -
- ))} + {rois.map((roi, i) => { + const attributes = roi.attributes + return ( +
+ ROI {roi.index} + {attributes.map((attr) => { + return ( +
+ {attr.name}: {attr.value} +
+ ) + })} +
+ + ) + })}
) } diff --git a/src/components/SlideViewer.tsx b/src/components/SlideViewer.tsx index 0ef1a4a..6be3beb 100644 --- a/src/components/SlideViewer.tsx +++ b/src/components/SlideViewer.tsx @@ -60,13 +60,12 @@ const DEFAULT_ROI_RADIUS: number = 5 const DEFAULT_ANNOTATION_OPACITY = 0.4 const DEFAULT_ANNOTATION_STROKE_COLOR = [0, 0, 0] const DEFAULT_ANNOTATION_COLOR_PALETTE = [ - [54, 162, 235], - [181, 65, 98], - [75, 192, 192], - [255, 158, 64], - [153, 102, 254], - [255, 205, 86], - [200, 203, 207] + [255, 0, 0], + [0, 255, 0], + [0, 0, 255], + [255, 255, 0], + [0, 255, 255], + [0, 0, 0] ] const _buildKey = (concept: { @@ -397,8 +396,7 @@ interface SlideViewerState { isAnnotationModalVisible: boolean isSelectedRoiModalVisible: boolean isHoveredRoiTooltipVisible: boolean - hoveredRoi?: dmv.roi.ROI - hoveredRoiAttributes: Array<{ name: string, value: string }> + hoveredRoiAttributes: Array<{index: number, roiUid: string, attributes: Array<{ name: string, value: string }>}> hoveredRoiTooltipX: number hoveredRoiTooltipY: number isReportModalVisible: boolean @@ -447,6 +445,10 @@ class SlideViewer extends React.Component { private labelViewer?: dmv.viewer.LabelImageViewer + private hoveredRois = [] as dmv.roi.ROI[] + + private lastPixel = [0, 0] as [number, number] + private readonly defaultRoiStyle: dmv.viewer.ROIStyleOptions = { stroke: { color: DEFAULT_ROI_STROKE_COLOR, @@ -471,6 +473,7 @@ class SlideViewer extends React.Component { [annotationUID: string]: { opacity: number color: number[] + contourOnly: boolean } } = {} @@ -1471,62 +1474,94 @@ class SlideViewer extends React.Component { } } - setHoveredRoiAttributes = (hoveredRoi: dmv.roi.ROI): void => { - const attributes: Array<{ name: string, value: string }> = [] - hoveredRoi.evaluations.forEach(( - item: ( - dcmjs.sr.valueTypes.TextContentItem | - dcmjs.sr.valueTypes.CodeContentItem - ) - ) => { - const nameValue = item.ConceptNameCodeSequence[0].CodeValue - const nameMeaning = item.ConceptNameCodeSequence[0].CodeMeaning - const name = `${nameMeaning}` - if (item.ValueType === dcmjs.sr.valueTypes.ValueTypes.CODE) { - const codeContentItem = item as dcmjs.sr.valueTypes.CodeContentItem - const valueMeaning = codeContentItem.ConceptCodeSequence[0].CodeMeaning - // For consistency with Segment and Annotation Group - if (nameValue === '276214006') { - attributes.push({ - name: 'Property category', - value: `${valueMeaning}` - }) - } else if (nameValue === '121071') { - attributes.push({ - name: 'Property type', - value: `${valueMeaning}` - }) - } else if (nameValue === '111001') { - attributes.push({ - name: 'Algorithm Name', - value: `${valueMeaning}` - }) - } else { + setHoveredRoiAttributes = (hoveredRois: dmv.roi.ROI[]): void => { + const rois = this.volumeViewer.getAllROIs() + const result = hoveredRois.map((roi) => { + const attributes: Array<{ name: string, value: string }> = [] + const evaluations = roi.evaluations + evaluations.forEach(( + item: ( + dcmjs.sr.valueTypes.TextContentItem | + dcmjs.sr.valueTypes.CodeContentItem + ) + ) => { + const nameValue = item.ConceptNameCodeSequence[0].CodeValue + const nameMeaning = item.ConceptNameCodeSequence[0].CodeMeaning + const name = `${nameMeaning}` + if (item.ValueType === dcmjs.sr.valueTypes.ValueTypes.CODE) { + const codeContentItem = item as dcmjs.sr.valueTypes.CodeContentItem + const valueMeaning = codeContentItem.ConceptCodeSequence[0].CodeMeaning + // For consistency with Segment and Annotation Group + if (nameValue === '276214006') { + attributes.push({ + name: 'Property category', + value: `${valueMeaning}` + }) + } else if (nameValue === '121071') { + attributes.push({ + name: 'Property type', + value: `${valueMeaning}` + }) + } else if (nameValue === '111001') { + attributes.push({ + name: 'Algorithm Name', + value: `${valueMeaning}` + }) + } else { + attributes.push({ + name: name, + value: `${valueMeaning}` + }) + } + } else if (item.ValueType === dcmjs.sr.valueTypes.ValueTypes.TEXT) { + const textContentItem = item as dcmjs.sr.valueTypes.TextContentItem attributes.push({ name: name, - value: `${valueMeaning}` + value: textContentItem.TextValue }) } - } else if (item.ValueType === dcmjs.sr.valueTypes.ValueTypes.TEXT) { - const textContentItem = item as dcmjs.sr.valueTypes.TextContentItem - attributes.push({ - name: name, - value: textContentItem.TextValue - }) - } - }) + }) - this.setState({ hoveredRoiAttributes: attributes }) + const index = (rois.findIndex((r) => r.uid === roi.uid) ?? 0) + 1 + return { index, roiUid: roi.uid, attributes } + }, [] as Array) + + this.setState({ hoveredRoiAttributes: result }) + } + + clearHoveredRois = (): void => { + this.hoveredRois = [] as any + } + + getUniqueHoveredRois = (newRoi: dmv.roi.ROI | null): dmv.roi.ROI[] => { + if (newRoi == null) { + return [] + } + const allRois = [...this.hoveredRois, newRoi] + const uniqueIds = Array.from(new Set(allRois.map(roi => roi.uid))) + // @ts-expect-error + return uniqueIds.map(id => allRois.find(roi => roi.uid === id)).filter(roi => roi !== undefined) + } + + isSamePixelAsLast = (event: any): boolean => { + return event.clientX === this.lastPixel[0] && event.clientY === this.lastPixel[1] } onPointerMove = (event: CustomEventInit): void => { const { feature: hoveredRoi, event: evt } = event.detail.payload - if (hoveredRoi != null) { - const originalEvent = evt.originalEvent - this.setHoveredRoiAttributes(hoveredRoi) + const originalEvent = evt.originalEvent + + if (!this.isSamePixelAsLast(originalEvent)) { + this.lastPixel = [originalEvent.clientX, originalEvent.clientY] + this.clearHoveredRois() + } + + this.hoveredRois = this.getUniqueHoveredRois(hoveredRoi) + + if (this.hoveredRois.length > 0) { + this.setHoveredRoiAttributes(this.hoveredRois) this.setState({ isHoveredRoiTooltipVisible: true, - hoveredRoi, hoveredRoiTooltipX: originalEvent.clientX, hoveredRoiTooltipY: originalEvent.clientY }) @@ -1538,26 +1573,21 @@ class SlideViewer extends React.Component { } onRoiSelected = (event: CustomEventInit): void => { - const selectedRoi = event.detail.payload as dmv.roi.ROI - if (selectedRoi != null) { - console.debug(`selected ROI "${selectedRoi.uid}"`) - this.volumeViewer.setROIStyle(selectedRoi.uid, this.selectedRoiStyle) - const key = _getRoiKey(selectedRoi) - this.volumeViewer.getAllROIs().forEach((roi) => { - if (roi.uid !== selectedRoi.uid) { - this.volumeViewer.setROIStyle(roi.uid, this.getRoiStyle(key)) - } - }) - this.setState({ - selectedRoiUIDs: new Set([selectedRoi.uid]), - selectedRoi: selectedRoi - }) - } else { + const selectedRoi = event.detail.payload as dmv.roi.ROI | null + if (selectedRoi == null) { this.setState({ selectedRoiUIDs: new Set(), selectedRoi: undefined }) + return } + + console.debug(`selected ROI "${selectedRoi.uid}"`) + const oldSelectedRois = Array.from(this.state.selectedRoiUIDs) + this.setState({ + selectedRoiUIDs: new Set([...oldSelectedRois, selectedRoi.uid]), + selectedRoi: selectedRoi + }) } handleRoiSelectionCancellation (): void { @@ -2522,10 +2552,11 @@ class SlideViewer extends React.Component { styleOptions: { opacity?: number color?: number[] + contourOnly: boolean }): dmv.viewer.ROIStyleOptions { const opacity = styleOptions.opacity ?? DEFAULT_ANNOTATION_OPACITY const strokeColor = styleOptions.color ?? DEFAULT_ANNOTATION_STROKE_COLOR - const fillColor = strokeColor.map((c) => Math.min(c + 25, 255)) + const fillColor = styleOptions.contourOnly ? [0, 0, 0, 0] : strokeColor.map((c) => Math.min(c + 25, 255)) const style = _formatRoiStyle({ fill: { color: [...fillColor, opacity] }, stroke: { color: [...strokeColor, opacity] }, @@ -2539,6 +2570,7 @@ class SlideViewer extends React.Component { styleOptions: { opacity: number color: number[] + contourOnly: boolean } }): void { console.log(`change style of ROI ${uid}`) @@ -3359,7 +3391,8 @@ class SlideViewer extends React.Component { ] this.defaultAnnotationStyles[annotation.uid] = { color, - opacity: DEFAULT_ANNOTATION_OPACITY + opacity: DEFAULT_ANNOTATION_OPACITY, + contourOnly: false } as any this.roiStyles[key] = this.generateRoiStyle( @@ -3814,7 +3847,7 @@ class SlideViewer extends React.Component { ) : (