DICOMZero.js 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. class DICOMZero {
  2. constructor(options={}) {
  3. this.status = options.status || function() {};
  4. this.reset();
  5. }
  6. reset() {
  7. this.mappingLog = [];
  8. this.dataTransfer = undefined;
  9. this.datasets = [];
  10. this.readers = [];
  11. this.arrayBuffers = [];
  12. this.files = [];
  13. this.fileIndex = 0;
  14. this.context = {patients: []};
  15. }
  16. static datasetFromArrayBuffer(arrayBuffer) {
  17. let dicomData = dcmjs.data.DicomMessage.readFile(arrayBuffer);
  18. let dataset = dcmjs.data.DicomMetaDictionary.naturalizeDataset(dicomData.dict);
  19. dataset._meta = dcmjs.data.DicomMetaDictionary.namifyDataset(dicomData.meta);
  20. return(dataset);
  21. }
  22. // return a function to use as the 'onload' callback for the file reader.
  23. // The function takes a progress event argument and it knows (from this class instance)
  24. // when all files have been read so it can invoke the doneCallback when all
  25. // have been read.
  26. getReadDICOMFunction(doneCallback, statusCallback) {
  27. statusCallback = statusCallback || console.log;
  28. return progressEvent => {
  29. let reader = progressEvent.target;
  30. let arrayBuffer = reader.result;
  31. this.arrayBuffers.push(arrayBuffer);
  32. let dicomData;
  33. try {
  34. dicomData = dcmjs.data.DicomMessage.readFile(arrayBuffer);
  35. let dataset = dcmjs.data.DicomMetaDictionary.naturalizeDataset(dicomData.dict);
  36. dataset._meta = dcmjs.data.DicomMetaDictionary.namifyDataset(dicomData.meta);
  37. this.datasets.push(dataset);
  38. } catch (error) {
  39. console.error(error);
  40. statusCallback("skipping non-dicom file");
  41. }
  42. let readerIndex = this.readers.indexOf(reader);
  43. if (readerIndex < 0) {
  44. reject("Logic error: Unexpected reader!");
  45. } else {
  46. this.readers.splice(readerIndex, 1); // remove the reader
  47. }
  48. if (this.fileIndex === this.dataTransfer.files.length) {
  49. statusCallback(`Normalizing...`);
  50. try {
  51. this.multiframe = dcmjs.normalizers.Normalizer.normalizeToDataset(this.datasets);
  52. } catch (e) {
  53. console.error('Could not convert to multiframe');
  54. console.error(e);
  55. }
  56. if (this.multiframe.SOPClassUID == dcmjs.data.DicomMetaDictionary.sopClassUIDsByName['Segmentation']){
  57. statusCallback(`Creating segmentation...`);
  58. try {
  59. this.seg = new dcmjs.derivations.Segmentation([this.multiframe]);
  60. statusCallback(`Created ${this.multiframe.NumberOfFrames} frame multiframe object and segmentation.`);
  61. } catch (e) {
  62. console.error('Could not create segmentation');
  63. console.error(e);
  64. }
  65. } else if (this.multiframe.SOPClassUID == dcmjs.data.DicomMetaDictionary.sopClassUIDsByName['ParametricMapStorage']){
  66. statusCallback(`Creating parametric map...`);
  67. try {
  68. this.pm = new dcmjs.derivations.ParametricMap([this.multiframe]);
  69. statusCallback(`Created ${this.multiframe.NumberOfFrames} frame multiframe object and parametric map.`);
  70. } catch (e) {
  71. console.error('Could not create parametric map');
  72. console.error(e);
  73. }
  74. }
  75. doneCallback();
  76. } else {
  77. statusCallback(`Reading... (${this.fileIndex+1}).`);
  78. this.readOneFile(doneCallback, statusCallback);
  79. }
  80. };
  81. }
  82. // Used for file selection button or drop of file list
  83. readOneFile(doneCallback, statusCallback) {
  84. let file = this.dataTransfer.files[this.fileIndex];
  85. this.fileIndex++;
  86. let reader = new FileReader();
  87. reader.onload = this.getReadDICOMFunction(doneCallback, statusCallback);
  88. reader.readAsArrayBuffer(file);
  89. this.files.push(file);
  90. this.readers.push(reader);
  91. }
  92. handleDataTransferFileAsDataset(file, options={}) {
  93. options.doneCallback = options.doneCallback || function(){};
  94. let reader = new FileReader();
  95. reader.onload = (progressEvent) => {
  96. let dataset = DICOMZero.datasetFromArrayBuffer(reader.result);
  97. options.doneCallback(dataset);
  98. }
  99. reader.readAsArrayBuffer(file);
  100. }
  101. extractDatasetFromZipArrayBuffer(arrayBuffer) {
  102. this.status(`Extracting ${this.datasets.length} of ${this.expectedDICOMFileCount}...`);
  103. this.datasets.push(DICOMZero.datasetFromArrayBuffer(arrayBuffer));
  104. if (this.datasets.length == this.expectedDICOMFileCount) {
  105. this.status(`Finished extracting`);
  106. this.zipFinishCallback();
  107. }
  108. };
  109. handleZip(zip) {
  110. this.zip = zip;
  111. this.expectedDICOMFileCount = 0;
  112. Object.keys(zip.files).forEach(fileKey => {
  113. this.status(`Considering ${fileKey}...`);
  114. if (fileKey.endsWith('.dcm')) {
  115. this.expectedDICOMFileCount += 1;
  116. zip.files[fileKey].async('arraybuffer').then(this.extractDatasetFromZipArrayBuffer.bind(this));
  117. }
  118. });
  119. }
  120. extractFromZipArrayBuffer(arrayBuffer, finishCallback=function(){}) {
  121. this.zipFinishCallback = finishCallback;
  122. this.status("Extracting from zip...");
  123. JSZip.loadAsync(arrayBuffer)
  124. .then(this.handleZip.bind(this));
  125. }
  126. organizeDatasets() {
  127. this.datasets.forEach(dataset => {
  128. let patientName = dataset.PatientName;
  129. let studyTag = dataset.StudyDate + ": " + dataset.StudyDescription;
  130. let seriesTag = dataset.SeriesNumber + ": " + dataset.SeriesDescription;
  131. let patientNames = this.context.patients.map(patient => patient.name);
  132. let patientIndex = patientNames.indexOf(dataset.PatientName);
  133. if (patientIndex == -1) {
  134. this.context.patients.push({
  135. name: dataset.PatientName,
  136. id: this.context.patients.length,
  137. studies: {}
  138. });
  139. }
  140. let studyNames; // TODO - finish organizing
  141. });
  142. }
  143. }