Press n or j to go to the next uncovered block, b, p or k for the previous block.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 | 1x 1x 1x 1x 5x 5x 5x 5x 1x 1x 5x 5x 4x 5x 1x 1x 1x 1x 1x 1x 1x | import { vec3 } from 'gl-matrix'; import { metaData, getConfiguration } from '@cornerstonejs/core'; import type { Types } from '@cornerstonejs/core'; type SortedImageIdsItem = { zSpacing: number; origin: Types.Point3; sortedImageIds: Array<string>; }; /** * Given an array of imageIds, sort them based on their imagePositionPatient, and * also returns the spacing between images and the origin of the reference image * * @param imageIds - array of imageIds * @param scanAxisNormal - [x, y, z] array or gl-matrix vec3 * * @returns The sortedImageIds, zSpacing, and origin of the first image in the series. */ export default function sortImageIdsAndGetSpacing( imageIds: Array<string>, scanAxisNormal: vec3 // Get gl matrix types? ): SortedImageIdsItem { const { imagePositionPatient: referenceImagePositionPatient } = metaData.get( 'imagePlaneModule', imageIds[0] ); const refIppVec = vec3.create(); // Check if we are using wadouri scheme const usingWadoUri = imageIds[0].split(':')[0] === 'wadouri'; vec3.set( refIppVec, referenceImagePositionPatient[0], referenceImagePositionPatient[1], referenceImagePositionPatient[2] ); let sortedImageIds: string[]; let zSpacing: number; function getDistance(imageId: string) { const { imagePositionPatient } = metaData.get('imagePlaneModule', imageId); const positionVector = vec3.create(); vec3.sub( positionVector, referenceImagePositionPatient, imagePositionPatient ); return vec3.dot(positionVector, scanAxisNormal); } /** * If we are using wadors and so have all image metadata cached ahead of time, * then sort by image position in 3D space, and calculate average slice * spacing from the entire volume. If not, then use the sampled images (1st * and middle) to calculate slice spacing, and use the provided imageId order. * Correct sorting must be done ahead of time. */ Eif (!usingWadoUri) { const distanceImagePairs = imageIds.map((imageId) => { const distance = getDistance(imageId); return { distance, imageId, }; }); distanceImagePairs.sort((a, b) => b.distance - a.distance); sortedImageIds = distanceImagePairs.map((a) => a.imageId); const numImages = distanceImagePairs.length; // Calculated average spacing. // We would need to resample if these are not similar. // It should be up to the host app to do this if it needed to. zSpacing = Math.abs( distanceImagePairs[numImages - 1].distance - distanceImagePairs[0].distance ) / (numImages - 1); } else { // Using wadouri, so we have only prefetched the first, middle, and last // images for metadata. Assume initial imageId array order is pre-sorted, // but check orientation. const prefetchedImageIds = [ imageIds[0], imageIds[Math.floor(imageIds.length / 2)], ]; sortedImageIds = imageIds; const firstImageDistance = getDistance(prefetchedImageIds[0]); const middleImageDistance = getDistance(prefetchedImageIds[1]); if (firstImageDistance - middleImageDistance < 0) { sortedImageIds.reverse(); } // Calculate average spacing between the first and middle prefetched images, // otherwise fall back to DICOM `spacingBetweenSlices` const metadataForMiddleImage = metaData.get( 'imagePlaneModule', prefetchedImageIds[1] ); if (!metadataForMiddleImage) { throw new Error('Incomplete metadata required for volume construction.'); } const positionVector = vec3.create(); vec3.sub( positionVector, referenceImagePositionPatient, metadataForMiddleImage.imagePositionPatient ); const distanceBetweenFirstAndMiddleImages = vec3.dot( positionVector, scanAxisNormal ); zSpacing = Math.abs(distanceBetweenFirstAndMiddleImages) / Math.floor(imageIds.length / 2); } const { imagePositionPatient: origin, sliceThickness } = metaData.get( 'imagePlaneModule', sortedImageIds[0] ); const { strictZSpacingForVolumeViewport } = getConfiguration().rendering; // We implemented these lines for multiframe dicom files that does not have // position for each frame, leading to incorrect calculation of zSpacing = 0 // If possible, we use the sliceThickness, but we warn about this dicom file // weirdness. If sliceThickness is not available, we set to 1 just to render Iif (zSpacing === 0 && !strictZSpacingForVolumeViewport) { if (sliceThickness) { console.log('Could not calculate zSpacing. Using sliceThickness'); zSpacing = sliceThickness; } else { console.log( 'Could not calculate zSpacing. The VolumeViewport visualization is compromised. Setting zSpacing to 1 to render' ); zSpacing = 1; } } const result: SortedImageIdsItem = { zSpacing, origin, sortedImageIds, }; return result; } |