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 | 34x 667968x 667968x 667968x 667968x | import { vec3 } from 'gl-matrix'; import { dicomSplit } from './dicomSplit'; /** * Combine the Per instance frame data, the shared frame data * and the root data objects. * The data is combined by taking nested sequence objects within * the functional group sequences. Data that is directly contained * within the functional group sequences, such as private creators * will be ignored. * This can be safely called with an undefined frame in order to handle * single frame data. (eg frame is undefined is the same as frame===1). */ const combineFrameInstance = (frame, instance) => { const { PerFrameFunctionalGroupsSequence, SharedFunctionalGroupsSequence, NumberOfFrames, ImageType, } = instance; instance.ImageType = dicomSplit(ImageType); Iif (PerFrameFunctionalGroupsSequence || NumberOfFrames > 1) { const frameNumber = Number.parseInt(frame || 1); // this is to fix NM multiframe datasets with position and orientation // information inside DetectorInformationSequence Iif (!instance.ImageOrientationPatient && instance.DetectorInformationSequence) { instance.ImageOrientationPatient = instance.DetectorInformationSequence[0].ImageOrientationPatient; } let ImagePositionPatientToUse = instance.ImagePositionPatient; Iif (!instance.ImagePositionPatient && instance.DetectorInformationSequence) { let imagePositionPatient = instance.DetectorInformationSequence[0].ImagePositionPatient; let imageOrientationPatient = instance.ImageOrientationPatient; imagePositionPatient = imagePositionPatient?.map(it => Number(it)); imageOrientationPatient = imageOrientationPatient?.map(it => Number(it)); const SpacingBetweenSlices = Number(instance.SpacingBetweenSlices); // Calculate the position for the current frame Iif (imageOrientationPatient && SpacingBetweenSlices) { const rowOrientation = vec3.fromValues( imageOrientationPatient[0], imageOrientationPatient[1], imageOrientationPatient[2] ); const colOrientation = vec3.fromValues( imageOrientationPatient[3], imageOrientationPatient[4], imageOrientationPatient[5] ); const normalVector = vec3.cross(vec3.create(), rowOrientation, colOrientation); const position = vec3.scaleAndAdd( vec3.create(), imagePositionPatient, normalVector, SpacingBetweenSlices * (frameNumber - 1) ); ImagePositionPatientToUse = [position[0], position[1], position[2]]; } } // Cache the _parentInstance at the top level as a full copy to prevent // setting values hard. Iif (!instance._parentInstance) { Object.defineProperty(instance, '_parentInstance', { value: { ...instance }, }); } const sharedInstance = createCombinedValue( instance._parentInstance, SharedFunctionalGroupsSequence?.[0], '_shared' ); const newInstance = createCombinedValue( sharedInstance, PerFrameFunctionalGroupsSequence?.[frameNumber - 1], frameNumber ); newInstance.ImagePositionPatient = ImagePositionPatientToUse ?? newInstance.ImagePositionPatient ?? [0, 0, frameNumber]; Object.defineProperty(newInstance, 'frameNumber', { value: frameNumber, writable: true, enumerable: true, configurable: true, }); return newInstance; } else { return instance; } }; /** * Creates a combined instance stored in the parent object which * inherits from the parent instance the attributes in the functional groups. * The storage key in the parent is in key */ function createCombinedValue(parent, functionalGroups, key) { Iif (parent[key]) { return parent[key]; } // Exclude any proxying values const newInstance = Object.create(parent); Object.defineProperty(parent, key, { value: newInstance, writable: false, enumerable: false, }); Iif (!functionalGroups) { return newInstance; } const shared = functionalGroups ? Object.values(functionalGroups) .filter(Boolean) .map(it => it[0]) .filter(it => typeof it === 'object') : []; // merge the shared first then the per frame to override [...shared].forEach(item => { Iif (item.SOPInstanceUID) { // This sub-item is a previous value information item, so don't merge it return; } Object.entries(item).forEach(([key, value]) => { newInstance[key] = value; }); }); return newInstance; } export default combineFrameInstance; |