From a78dc001f25369147485c3868e5664cc794c9e75 Mon Sep 17 00:00:00 2001 From: Gitanjali <112136540+GitanjaliChhetri@users.noreply.github.com> Date: Thu, 9 May 2024 14:44:47 -0600 Subject: [PATCH] refactor(viewer.js): Add errorinterceptor argument to propagate error to the top level app (#85) * add-error-interceptor * Apply suggestions from code review Co-authored-by: Markus D. Herrmann * Update documentation comments and add Error variables * add custom error class * fix linting errors * dummy commit to trigger actions workflow * update file name * Update PR * Update PR: Update signature * Lint * Revert "fix: Setup the dynamic import so it is separate from other packages (#113)" This reverts commit 28349445228ab2f4b67504c0f06f21710ebe495a. * Fix enum * Revert "Revert "fix: Setup the dynamic import so it is separate from other packages (#113)"" This reverts commit f5abca9265ced1aaf8075fac88b8780396233298. --------- Co-authored-by: Markus D. Herrmann Co-authored-by: Igor Octaviano --- src/customError.js | 15 +++ src/viewer.js | 317 +++++++++++++++++++++++++++++++++++---------- 2 files changed, 264 insertions(+), 68 deletions(-) create mode 100644 src/customError.js diff --git a/src/customError.js b/src/customError.js new file mode 100644 index 0000000..6a78a5f --- /dev/null +++ b/src/customError.js @@ -0,0 +1,15 @@ +const errorTypes = { + VISUALIZATION: 'Visualization', + ENCODINGANDDECODING: 'EncodingDecoding' +} + +class CustomError extends Error { + constructor (type, message) { + super() + this.message = message + this.stack = new Error().stack + this.type = type + } +} + +export { errorTypes, CustomError } diff --git a/src/viewer.js b/src/viewer.js index 02309ba..9467b08 100644 --- a/src/viewer.js +++ b/src/viewer.js @@ -39,6 +39,7 @@ import { ZoomSlider, Zoom } from 'ol/control' import { getCenter, getHeight, getWidth } from 'ol/extent' import { defaults as defaultInteractions } from 'ol/interaction' import dcmjs from 'dcmjs' +import { CustomError, errorTypes } from './customError' import _ from 'lodash' import { @@ -746,6 +747,8 @@ class VolumeImageViewer { * the application * @param {number[]} [options.highlightColor=[140, 184, 198]] - Color that * should be used to highlight things that get selected by the user + * @param {errorInterceptor} [options.errorInterceptor] - Callback for + * intercepting errors * @param {number[]} [options.mapViewResolutions] Map's view list of * resolutions. If not passed, the tile grid resolution will be used. */ @@ -757,21 +760,35 @@ class VolumeImageViewer { this[_clients].default = this[_options].client } else { if (this[_options].clientMapping == null) { - throw new Error( - 'Either option "client" or option "clientMapping" must be provided.' + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, + 'Either option "client" or option "clientMapping" must be provided .' ) + throw this[_options].errorInterceptor(error) || error } if (!(typeof this[_options].clientMapping === 'object')) { - throw new Error('Option "clientMapping" must be an object.') + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, + 'Option "clientMapping" must be an object.' + ) + throw this[_options].errorInterceptor(error) || error } if (this[_options].clientMapping.default == null) { - throw new Error('Option "clientMapping" must contain "default" key.') + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, + 'Option "clientMapping" must contain "default" key.' + ) + throw this[_options].errorInterceptor(error) || error } for (const key in this[_options].clientMapping) { this[_clients][key] = this[_options].clientMapping[key] } } + if (this[_options].errorInterceptor == null) { + this[_options].errorInterceptor = error => error + } + if (this[_options].debug == null) { this[_options].debug = false } else { @@ -825,15 +842,27 @@ class VolumeImageViewer { }) if (this[_options].metadata.constructor.name !== 'Array') { - throw new Error('Input metadata must be an array.') + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, + 'Input metadata must be an array.' + ) + throw this[_options].errorInterceptor(error) || error } if (this[_options].metadata.length === 0) { - throw new Error('Input metadata array is empty.') + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, + 'Input metadata array is empty.' + ) + throw this[_options].errorInterceptor(error) || error } if (this[_options].metadata.some((item) => typeof item !== 'object')) { - throw new Error('Input metadata must be an array of objects.') + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, + 'Input metadata must be an array of objects.' + ) + throw this[_options].errorInterceptor(error) || error } // We also accept metadata in raw JSON format for backwards compatibility @@ -882,13 +911,25 @@ class VolumeImageViewer { const numChannels = monochromeOpticalPathIdentifiers.length const numColorImages = colorOpticalPathIdentifiers.length if (numChannels === 0 && numColorImages === 0) { - throw new Error('Could not find any channels or color images.') + const error = new CustomError( + errorTypes.VISUALIZATION, + 'Could not find any channels or color images.' + ) + throw this[_options].errorInterceptor(error) || error } if (numChannels > 0 && numColorImages > 0) { - throw new Error('Found both channels and color images.') + const error = new CustomError( + errorTypes.VISUALIZATION, + 'Found both channels and color images.' + ) + throw this[_options].errorInterceptor(error) || error } if (numColorImages > 1) { - throw new Error('Found more than one color image.') + const error = new CustomError( + errorTypes.VISUALIZATION, + 'Found more than one color image.' + ) + throw this[_options].errorInterceptor(error) || error } /* @@ -1079,10 +1120,12 @@ class VolumeImageViewer { this[_pyramid] ) if (!areImagePyramidsEqual) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, `Pyramid of optical path "${opticalPathIdentifier}" ` + 'is different from reference pyramid.' ) + throw this[_options].errorInterceptor(error) || error } const source = new DataTileSource({ @@ -1543,10 +1586,12 @@ class VolumeImageViewer { setOpticalPathStyle (opticalPathIdentifier, styleOptions = {}) { const opticalPath = this[_opticalPaths][opticalPathIdentifier] if (opticalPath === undefined) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot set optical path style. Could not find optical path ' + `"${opticalPathIdentifier}".` ) + throw this[_options].errorInterceptor(error) || error } if (Object.entries(styleOptions).length === 0) { @@ -1651,10 +1696,12 @@ class VolumeImageViewer { getOpticalPathDefaultStyle (opticalPathIdentifier) { const opticalPath = this[_opticalPaths][opticalPathIdentifier] if (opticalPath == null) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot get default style of optical path. ' + `Could not find optical path "${opticalPathIdentifier}".` ) + throw this[_options].errorInterceptor(error) || error } if (opticalPath.opticalPath.isMonochromatic) { if (opticalPath.defaultStyle.paletteColorLookupTable) { @@ -1682,10 +1729,12 @@ class VolumeImageViewer { getOpticalPathStyle (opticalPathIdentifier) { const opticalPath = this[_opticalPaths][opticalPathIdentifier] if (opticalPath == null) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot get style of optical path. ' + `Could not find optical path "${opticalPathIdentifier}".` ) + throw this[_options].errorInterceptor(error) || error } if (opticalPath.opticalPath.isMonochromatic) { if (opticalPath.style.paletteColorLookupTable) { @@ -1714,10 +1763,12 @@ class VolumeImageViewer { getOpticalPathMetadata (opticalPathIdentifier) { const opticalPath = this[_opticalPaths][opticalPathIdentifier] if (opticalPath === undefined) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot get image metadata optical path. ' + `Could not find optical path "${opticalPathIdentifier}".` ) + throw this[_options].errorInterceptor(error) || error } return opticalPath.pyramid.metadata } @@ -1743,10 +1794,12 @@ class VolumeImageViewer { activateOpticalPath (opticalPathIdentifier) { const opticalPath = this[_opticalPaths][opticalPathIdentifier] if (opticalPath === undefined) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot activate optical path. Could not find optical path ' + `"${opticalPathIdentifier}".` ) + throw this[_options].errorInterceptor(error) || error } if (!this.isOpticalPathActive(opticalPathIdentifier)) { /* @@ -1774,10 +1827,12 @@ class VolumeImageViewer { deactivateOpticalPath (opticalPathIdentifier) { const opticalPath = this[_opticalPaths][opticalPathIdentifier] if (opticalPath === undefined) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot deactivate optical path. Could not find optical path ' + `"${opticalPathIdentifier}".` ) + throw this[_options].errorInterceptor(error) || error } if (!this.isOpticalPathActive(opticalPathIdentifier)) { return @@ -1829,10 +1884,12 @@ class VolumeImageViewer { showOpticalPath (opticalPathIdentifier, styleOptions = {}) { const opticalPath = this[_opticalPaths][opticalPathIdentifier] if (opticalPath === undefined) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot show optical path. Could not find optical path ' + `"${opticalPathIdentifier}".` ) + throw this[_options].errorInterceptor(error) || error } console.info(`show optical path ${opticalPathIdentifier}`) this.activateOpticalPath(opticalPathIdentifier) @@ -1868,10 +1925,12 @@ class VolumeImageViewer { hideOpticalPath (opticalPathIdentifier) { const opticalPath = this[_opticalPaths][opticalPathIdentifier] if (opticalPath === undefined) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot hide optical path. Could not find optical path ' + `"${opticalPathIdentifier}".` ) + throw this[_options].errorInterceptor(error) || error } console.info(`hide optical path ${opticalPathIdentifier}`) opticalPath.layer.setVisible(false) @@ -1887,10 +1946,12 @@ class VolumeImageViewer { isOpticalPathVisible (opticalPathIdentifier) { const opticalPath = this[_opticalPaths][opticalPathIdentifier] if (opticalPath === undefined) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot show optical path. Could not find optical path ' + `"${opticalPathIdentifier}".` ) + throw this[_options].errorInterceptor(error) || error } return opticalPath.layer.getVisible() } @@ -2208,7 +2269,11 @@ class VolumeImageViewer { */ navigate ({ level, position }) { if (level > this.numLevels) { - throw new Error('Argument "level" exceeds number of resolution levels.') + const error = new CustomError( + errorTypes.VISUALIZATION, + 'Argument "level" exceeds number of resolution levels.' + ) + throw this[_options].errorInterceptor(error) || error } let coordinates if (position != null) { @@ -2453,7 +2518,11 @@ class VolumeImageViewer { } catch (error) { const uid = feature.getId() this.removeROI(uid) - throw error + const roiError = new CustomError( + errorTypes.VISUALIZATION, + 'Unable to get ROI' + ) + this[_options].errorInterceptor(roiError) } const featureProperties = feature.getProperties() @@ -2784,7 +2853,11 @@ class VolumeImageViewer { console.debug(`get ROI ${uid}`) const feature = this[_drawingSource].getFeatureById(uid) if (feature == null) { - throw new Error(`Could not find a ROI with UID "${uid}".`) + const error = new CustomError( + errorTypes.VISUALIZATION, + `Could not find a ROI with UID "${uid}".` + ) + throw this[_options].errorInterceptor(error) || error } return this._getROIFromFeature( feature, @@ -2885,10 +2958,12 @@ class VolumeImageViewer { const frameOfReferenceUID = this[_pyramid].metadata.FrameOfReferenceUID if (roi.frameOfReferenceUID !== frameOfReferenceUID) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, `Frame of Reference UID of ROI ${roi.uid} does not match ` + 'Frame of Reference UID of source images.' ) + throw this[_options].errorInterceptor(error) || error } const geometry = _scoord3d2Geometry( @@ -2950,7 +3025,11 @@ class VolumeImageViewer { return feature.getId() === uid }) if (feature == null) { - throw new Error(`Could not find a ROI with UID "${uid}".`) + const error = new CustomError( + errorTypes.VISUALIZATION, + `Could not find a ROI with UID "${uid}".` + ) + throw this[_options].errorInterceptor(error) || error } const style = feature.getStyle() const stroke = style.getStroke() @@ -3104,10 +3183,12 @@ class VolumeImageViewer { addAnnotationGroups (metadata) { const refImage = this[_pyramid].metadata[0] if (refImage.FrameOfReferenceUID !== metadata.FrameOfReferenceUID) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Microscopy Bulk Simple Annotation instances must have the same ' + 'Frame of Reference UID as the corresponding source images.' ) + throw this[_options].errorInterceptor(error) || error } console.info( 'add annotation groups of Microscopy Bulk Simple Annotation instances ' + @@ -3130,10 +3211,12 @@ class VolumeImageViewer { item => item.AnnotationGroupUID === annotationGroupUID ) if (annotationGroupUID == null || annotationGroupMetadata == null) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Could not obtain information of annotation from ' + `annotation group "${annotationGroupUID}".` ) + throw this[_options].errorInterceptor(error) || error } if (annotationGroupMetadata.AnnotationPropertyCategoryCodeSequence != null) { @@ -3467,10 +3550,12 @@ class VolumeImageViewer { */ removeAnnotationGroup (annotationGroupUID) { if (!(annotationGroupUID in this[_annotationGroups])) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Cannot remove annotation group. ' + `Could not find annotation group "${annotationGroupUID}".` ) + throw this[_options].errorInterceptor(error) || error } const annotationGroup = this[_annotationGroups][annotationGroupUID] console.info(`remove annotation group ${annotationGroupUID}`) @@ -3500,10 +3585,12 @@ class VolumeImageViewer { */ showAnnotationGroup (annotationGroupUID, styleOptions = {}) { if (!(annotationGroupUID in this[_annotationGroups])) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Cannot show annotation group. ' + `Could not find annotation group "${annotationGroupUID}".` ) + throw this[_options].errorInterceptor(error) || error } const annotationGroup = this[_annotationGroups][annotationGroupUID] console.info(`show annotation group ${annotationGroupUID}`) @@ -3519,10 +3606,12 @@ class VolumeImageViewer { */ hideAnnotationGroup (annotationGroupUID) { if (!(annotationGroupUID in this[_annotationGroups])) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Cannot hide annotation group. ' + `Could not find annotation group "${annotationGroupUID}".` ) + throw this[_options].errorInterceptor(error) || error } const annotationGroup = this[_annotationGroups][annotationGroupUID] console.info(`hide annotation group ${annotationGroupUID}`) @@ -3537,10 +3626,12 @@ class VolumeImageViewer { */ isAnnotationGroupVisible (annotationGroupUID) { if (!(annotationGroupUID in this[_annotationGroups])) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Cannot determine if annotation group is visible. ' + `Could not find annotation group "${annotationGroupUID}".` ) + throw this[_options].errorInterceptor(error) || error } const annotationGroup = this[_annotationGroups][annotationGroupUID] return annotationGroup.layer.getVisible() @@ -3559,10 +3650,12 @@ class VolumeImageViewer { */ setAnnotationGroupStyle (annotationGroupUID, styleOptions = {}) { if (!(annotationGroupUID in this[_annotationGroups])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot set style of annotation group. ' + `Could not find annotation group "${annotationGroupUID}".` ) + throw this[_options].errorInterceptor(error) || error } const annotationGroup = this[_annotationGroups][annotationGroupUID] console.info( @@ -3584,10 +3677,12 @@ class VolumeImageViewer { return item.AnnotationGroupUID === annotationGroupUID }) if (groupItem == null) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot set style of annotation group. ' + `Could not find metadata of annotation group "${annotationGroupUID}".` ) + throw this[_options].errorInterceptor(error) || error } const markerType = 'circle' @@ -3612,11 +3707,13 @@ class VolumeImageViewer { return areCodedConceptsEqual(name, getContentItemNameCodedConcept(item)) }) if (measurementIndex == null) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot set style of annotation group. ' + `Could not find measurement "${name.CodeMeaning}" ` + `of annotation group "${annotationGroupUID}".` ) + throw this[_options].errorInterceptor(error) || error } const properties = source.getProperties() const key = `measurementValue${measurementIndex.toString()}` @@ -3696,10 +3793,12 @@ class VolumeImageViewer { */ getAnnotationGroupDefaultStyle (annotationGroupUID) { if (!(annotationGroupUID in this[_annotationGroups])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot get default style of annotation group. ' + `Could not find annotation group "${annotationGroupUID}".` ) + throw this[_options].errorInterceptor(error) || error } const annotationGroup = this[_annotationGroups][annotationGroupUID] return { @@ -3718,10 +3817,12 @@ class VolumeImageViewer { */ getAnnotationGroupStyle (annotationGroupUID) { if (!(annotationGroupUID in this[_annotationGroups])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot get style of annotation group. ' + `Could not find annotation group "${annotationGroupUID}".` ) + throw this[_options].errorInterceptor(error) || error } const annotationGroup = this[_annotationGroups][annotationGroupUID] return { @@ -3753,10 +3854,12 @@ class VolumeImageViewer { */ getAnnotationGroupMetadata (annotationGroupUID) { if (!(annotationGroupUID in this[_annotationGroups])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot get metadata of annotation group. ' + `Could not find annotation group "${annotationGroupUID}".` ) + throw this[_options].errorInterceptor(error) || error } const annotationGroup = this[_annotationGroups][annotationGroupUID] return annotationGroup.metadata @@ -3769,10 +3872,12 @@ class VolumeImageViewer { */ addSegments (metadata) { if (metadata.length === 0) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Metadata of Segmentation instances needs to be provided to ' + 'add segments.' ) + throw this[_options].errorInterceptor(error) || error } const refSegmentation = metadata[0] const refImage = this[_pyramid].metadata[0] @@ -3781,35 +3886,45 @@ class VolumeImageViewer { instance.TotalPixelMatrixColumns === undefined || instance.TotalPixelMatrixRows === undefined ) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Segmentation instances must contain attributes ' + '"Total Pixel Matrix Rows" and "Total Pixel Matrix Columns".' ) + throw this[_options].errorInterceptor(error) || error } if (refImage.FrameOfReferenceUID !== instance.FrameOfReferenceUID) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Segmentation instances must have the same Frame of Reference UID ' + 'as the corresponding source images.' ) + throw this[_options].errorInterceptor(error) || error } if (refSegmentation.FrameOfReferenceUID !== instance.FrameOfReferenceUID) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Segmentation instances must all have same Frame of Reference UID.' ) + throw this[_options].errorInterceptor(error) || error } if (refSegmentation.SeriesInstanceUID !== instance.SeriesInstanceUID) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Segmentation instances must all have same Series Instance UID.' ) + throw this[_options].errorInterceptor(error) || error } if ( refSegmentation.SegmentSequence.length !== instance.SegmentSequence.length ) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Segmentation instances must all contain the same number of items ' + 'in the Segment Sequence.' ) + throw this[_options].errorInterceptor(error) || error } }) console.info( @@ -3947,9 +4062,11 @@ class VolumeImageViewer { */ removeSegment (segmentUID) { if (!(segmentUID in this[_segments])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, `Cannot remove segment. Could not find segment "${segmentUID}".` ) + throw this[_options].errorInterceptor(error) || error } const segment = this[_segments][segmentUID] this[_map].removeLayer(segment.layer) @@ -3976,9 +4093,11 @@ class VolumeImageViewer { */ showSegment (segmentUID, styleOptions = {}) { if (!(segmentUID in this[_segments])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, `Cannot show segment. Could not find segment "${segmentUID}".` ) + throw this[_options].errorInterceptor(error) || error } const segment = this[_segments][segmentUID] console.info(`show segment ${segmentUID}`) @@ -4014,9 +4133,11 @@ class VolumeImageViewer { */ hideSegment (segmentUID) { if (!(segmentUID in this[_segments])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, `Cannot hide segment. Could not find segment "${segmentUID}".` ) + throw this[_options].errorInterceptor(error) || error } const segment = this[_segments][segmentUID] console.info(`hide segment ${segmentUID}`) @@ -4032,10 +4153,12 @@ class VolumeImageViewer { */ isSegmentVisible (segmentUID) { if (!(segmentUID in this[_segments])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot determine if segment is visible. ' + `Could not find segment "${segmentUID}".` ) + throw this[_options].errorInterceptor(error) || error } const segment = this[_segments][segmentUID] return segment.layer.getVisible() @@ -4050,10 +4173,12 @@ class VolumeImageViewer { */ setSegmentStyle (segmentUID, styleOptions = {}) { if (!(segmentUID in this[_segments])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot set style of segment. ' + `Could not find segment "${segmentUID}".` ) + throw this[_options].errorInterceptor(error) || error } const segment = this[_segments][segmentUID] @@ -4114,10 +4239,12 @@ class VolumeImageViewer { */ getSegmentDefaultStyle (segmentUID) { if (!(segmentUID in this[_segments])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot get default style of segment. ' + `Could not find segment "${segmentUID}".` ) + throw this[_options].errorInterceptor(error) || error } const segment = this[_segments][segmentUID] return { @@ -4135,10 +4262,12 @@ class VolumeImageViewer { */ getSegmentStyle (segmentUID) { if (!(segmentUID in this[_segments])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot get style of segment. ' + `Could not find segment "${segmentUID}".` ) + throw this[_options].errorInterceptor(error) || error } const segment = this[_segments][segmentUID] return { @@ -4156,10 +4285,12 @@ class VolumeImageViewer { */ getSegmentMetadata (segmentUID) { if (!(segmentUID in this[_segments])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot get image metadata of segment. ' + `Could not find segment "${segmentUID}".` ) + throw this[_options].errorInterceptor(error) || error } const segment = this[_segments][segmentUID] return segment.pyramid.metadata @@ -4185,10 +4316,12 @@ class VolumeImageViewer { */ addParameterMappings (metadata) { if (metadata.length === 0) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Metadata of Parametric Map instances needs to be provided to ' + 'add mappings.' ) + throw this[_options].errorInterceptor(error) || error } const refImage = this[_pyramid].metadata[0] @@ -4206,26 +4339,34 @@ class VolumeImageViewer { instance.TotalPixelMatrixColumns === undefined || instance.TotalPixelMatrixRows === undefined ) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Parametric Map instances must contain attributes ' + '"Total Pixel Matrix Rows" and "Total Pixel Matrix Columns".' ) + throw this[_options].errorInterceptor(error) || error } if (refImage.FrameOfReferenceUID !== instance.FrameOfReferenceUID) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Parametric Map instances must have the same Frame of Reference UID ' + 'as the corresponding source images.' ) + throw this[_options].errorInterceptor(error) || error } if (refParametricMap.FrameOfReferenceUID !== instance.FrameOfReferenceUID) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Parametric Map instances must all have same Frame of Reference UID.' ) + throw this[_options].errorInterceptor(error) || error } if (refParametricMap.SeriesInstanceUID !== instance.SeriesInstanceUID) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Parametric Map instances must all have same Series Instance UID.' ) + throw this[_options].errorInterceptor(error) || error } }) console.info( @@ -4252,10 +4393,12 @@ class VolumeImageViewer { const sharedFuncGroup = refInstance.SharedFunctionalGroupsSequence[0] const frameVOILUT = sharedFuncGroup.FrameVOILUTSequence[0] if (frameVOILUT === undefined) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'The Parametric Map image does not specify a shared frame ' + 'Value of Interest (VOI) lookup table (LUT).' ) + throw this[_options].errorInterceptor(error) || error } const windowCenter = frameVOILUT.WindowCenter const windowWidth = frameVOILUT.WindowWidth @@ -4279,11 +4422,13 @@ class VolumeImageViewer { mappingDescriptions.forEach((item, i) => { if (item.TrackingUID != null) { if (item.TrackingUID !== mappingUID) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, `Item #${i + 1} of Real World Value Mapping Sequence ` + `of frame #${index + 1} has unexpected Tracking UID. ` + 'All items must have the same unique identifier value.' ) + throw this[_options].errorInterceptor(error) || error } } let firstValueMapped = item.RealWorldValueFirstValueMapped @@ -4307,7 +4452,11 @@ class VolumeImageViewer { // TODO: include real world values in legend if (isNaN(range[0]) || isNaN(range[1])) { - throw new Error('Could not determine range of real world values.') + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, + 'Could not determine range of real world values.' + ) + throw this[_options].errorInterceptor(error) || error } let colormap @@ -4423,9 +4572,11 @@ class VolumeImageViewer { */ removeParameterMapping (mappingUID) { if (!(mappingUID in this[_mappings])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, `Cannot remove mapping. Could not find mapping "${mappingUID}".` ) + throw this[_options].errorInterceptor(error) || error } const mapping = this[_mappings][mappingUID] this[_map].removeLayer(mapping.layer) @@ -4453,9 +4604,11 @@ class VolumeImageViewer { */ showParameterMapping (mappingUID, styleOptions = {}) { if (!(mappingUID in this[_mappings])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, `Cannot show mapping. Could not find mapping "${mappingUID}".` ) + throw this[_options].errorInterceptor(error) || error } const mapping = this[_mappings][mappingUID] console.info(`show mapping ${mappingUID}`) @@ -4491,9 +4644,11 @@ class VolumeImageViewer { */ hideParameterMapping (mappingUID) { if (!(mappingUID in this[_mappings])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, `Cannot hide mapping. Could not find mapping "${mappingUID}".` ) + throw this[_options].errorInterceptor(error) || error } const mapping = this[_mappings][mappingUID] console.info(`hide mapping ${mappingUID}`) @@ -4509,10 +4664,12 @@ class VolumeImageViewer { */ isParameterMappingVisible (mappingUID) { if (!(mappingUID in this[_mappings])) { - throw new Error( + const error = new CustomError( + errorTypes.ENCODINGANDDECODING, 'Cannot determine if mapping is visible. ' + `Could not find mapping "${mappingUID}".` ) + throw this[_options].errorInterceptor(error) || error } const mapping = this[_mappings][mappingUID] return mapping.layer.getVisible() @@ -4528,10 +4685,12 @@ class VolumeImageViewer { */ setParameterMappingStyle (mappingUID, styleOptions = {}) { if (!(mappingUID in this[_mappings])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot set style of mapping. ' + `Could not find mapping "${mappingUID}".` ) + throw this[_options].errorInterceptor(error) || error } const mapping = this[_mappings][mappingUID] @@ -4606,10 +4765,12 @@ class VolumeImageViewer { */ getParameterMappingDefaultStyle (mappingUID) { if (!(mappingUID in this[_mappings])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot get default style of mapping. ' + `Could not find mapping "${mappingUID}".` ) + throw this[_options].errorInterceptor(error) || error } const mapping = this[_mappings][mappingUID] return { @@ -4627,10 +4788,12 @@ class VolumeImageViewer { */ getParameterMappingStyle (mappingUID) { if (!(mappingUID in this[_mappings])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot get style of mapping. ' + `Could not find mapping "${mappingUID}".` ) + throw this[_options].errorInterceptor(error) || error } const mapping = this[_mappings][mappingUID] return { @@ -4650,10 +4813,12 @@ class VolumeImageViewer { */ getParameterMappingMetadata (mappingUID) { if (!(mappingUID in this[_mappings])) { - throw new Error( + const error = new CustomError( + errorTypes.VISUALIZATION, 'Cannot get image metadata of mapping. ' + `Could not find mapping "${mappingUID}".` ) + throw this[_options].errorInterceptor(error) || error } const mapping = this[_mappings][mappingUID] return mapping.pyramid.metadata @@ -4696,6 +4861,10 @@ class _NonVolumeImageViewer { * should be included for correction of image colors. */ constructor (options) { + if (options.errorInterceptor == null) { + options.errorInterceptor = error => error + } + // We also accept metadata in raw JSON format for backwards compatibility if (options.metadata.SOPClassUID != null) { this[_metadata] = options.metadata @@ -4707,7 +4876,11 @@ class _NonVolumeImageViewer { const imageFlavor = this[_metadata].ImageType[2] if (imageFlavor === 'VOLUME') { - throw new Error('Viewer cannot render images of type VOLUME.') + const error = new CustomError( + errorTypes.VISUALIZATION, + 'Viewer cannot render images of type VOLUME.' + ) + options.errorInterceptor(error) } const resizeFactor = options.resizeFactor ? options.resizeFactor : 1 @@ -4890,6 +5063,10 @@ class OverviewImageViewer extends _NonVolumeImageViewer { * should be included for correction of image colors. */ constructor (options) { + if (options.errorInterceptor == null) { + options.errorInterceptor = error => error + } + if (options.orientation === undefined) { options.orientation = 'horizontal' } @@ -4918,6 +5095,10 @@ class LabelImageViewer extends _NonVolumeImageViewer { * should be included for correction of image colors */ constructor (options) { + if (options.errorInterceptor == null) { + options.errorInterceptor = error => error + } + if (options.orientation === undefined) { options.orientation = 'vertical' }