forked from e0404/matRad
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmatRad_calcQualityIndicators.m
148 lines (124 loc) · 5.94 KB
/
matRad_calcQualityIndicators.m
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
function qi = matRad_calcQualityIndicators(cst,pln,doseCube,refGy,refVol,param)
% %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
% matRad QI calculation
%
% call
% qi = matRad_calcQualityIndicators(cst,pln,doseCube,refGy,refVol,param)
%
% input
% cst: matRad cst struct
% pln: matRad pln struct
% doseCube: arbitrary doseCube (e.g. physicalDose)
% refGy: (optional) array of dose values used for V_XGy calculation
% default is [40 50 60]
% refVol:(optional) array of volumes (0-100) used for D_X calculation
% default is [2 5 95 98]
% NOTE: Call either both or none!
%
% output
% qi various quality indicators like CI, HI (for
% targets) and DX, VX within a structure set
%
% References
% van't Riet et. al., IJROBP, 1997 Feb 1;37(3):731-6.
% Kataria et. al., J Med Phys. 2012 Oct-Dec; 37(4)
%
% %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
% %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%
% Copyright 2016 the matRad development team.
%
% This file is part of the matRad project. It is subject to the license
% terms in the LICENSE file found in the top-level directory of this
% distribution and at https://github.com/e0404/matRad/LICENSES.txt. No part
% of the matRad project, including this file, may be copied, modified,
% propagated, or distributed except according to the terms contained in the
% LICENSE file.
%
% %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
if ~exist('refVol', 'var') || isempty(refVol)
refVol = [2 5 50 95 98];
end
if ~exist('refGy', 'var') || isempty(refGy)
refGy = floor(linspace(0,max(doseCube(:)),6)*10)/10;
end
if exist('param','var')
if ~isfield(param,'logLevel')
param.logLevel = 1;
end
else
param.logLevel = 1;
end
% calculate QIs per VOI
qi = struct;
for runVoi = 1:size(cst,1)
indices = cst{runVoi,4}{1};
numOfVoxels = numel(indices);
voiPrint = sprintf('%3d %20s',cst{runVoi,1},cst{runVoi,2}); %String that will print quality indicators
% get Dose, dose is sorted to simplify calculations
doseInVoi = sort(doseCube(indices));
if ~isempty(doseInVoi)
qi(runVoi).name = cst{runVoi,2};
% easy stats
qi(runVoi).mean = mean(doseInVoi);
qi(runVoi).std = std(doseInVoi);
qi(runVoi).max = doseInVoi(end);
qi(runVoi).min = doseInVoi(1);
voiPrint = sprintf('%s - Mean dose = %5.2f Gy +/- %5.2f Gy (Max dose = %5.2f Gy, Min dose = %5.2f Gy)\n%27s', ...
voiPrint,qi(runVoi).mean,qi(runVoi).std,qi(runVoi).max,qi(runVoi).min,' ');
DX = @(x) matRad_interp1(linspace(0,1,numOfVoxels),doseInVoi,(100-x)*0.01);
VX = @(x) numel(doseInVoi(doseInVoi >= x)) / numOfVoxels;
% create VX and DX struct fieldnames at runtime and fill
for runDX = 1:numel(refVol)
qi(runVoi).(strcat('D_',num2str(refVol(runDX)))) = DX(refVol(runDX));
voiPrint = sprintf('%sD%d%% = %5.2f Gy, ',voiPrint,refVol(runDX),DX(refVol(runDX)));
end
voiPrint = sprintf('%s\n%27s',voiPrint,' ');
for runVX = 1:numel(refGy)
sRefGy = num2str(refGy(runVX),3);
qi(runVoi).(['V_' strrep(sRefGy,'.','_') 'Gy']) = VX(refGy(runVX));
voiPrint = sprintf(['%sV' sRefGy 'Gy = %6.2f%%, '],voiPrint,VX(refGy(runVX))*100);
end
voiPrint = sprintf('%s\n%27s',voiPrint,' ');
% if current voi is a target -> calculate homogeneity and conformity
if strcmp(cst{runVoi,3},'TARGET') > 0
% loop over target objectives and get the lowest dose objective
referenceDose = inf;
for runObjective = 1:numel(cst{runVoi,6})
% check if this is an objective that penalizes underdosing
if strcmp(cst{runVoi,6}(runObjective).type,'square deviation') > 0 || strcmp(cst{runVoi,6}(runObjective).type,'square underdosing') > 0
referenceDose = (min(cst{runVoi,6}(runObjective).dose,referenceDose))/pln.numOfFractions;
end
end
if referenceDose == inf
voiPrint = sprintf('%s%s',voiPrint,'Warning: target has no objective that penalizes underdosage, ');
else
StringReferenceDose = regexprep(num2str(round(referenceDose*100)/100),'\D','_');
% Conformity Index, fieldname contains reference dose
VTarget95 = sum(doseInVoi >= 0.95*referenceDose); % number of target voxels recieving dose >= 0.95 dPres
VTreated95 = sum(doseCube(:) >= 0.95*referenceDose); %number of all voxels recieving dose >= 0.95 dPres ("treated volume")
qi(runVoi).(['CI_' StringReferenceDose 'Gy']) = VTarget95^2/(numOfVoxels * VTreated95);
% Homogeneity Index (one out of many), fieldname contains reference dose
qi(runVoi).(['HI_' StringReferenceDose 'Gy']) = (DX(5) - DX(95))/referenceDose * 100;
voiPrint = sprintf('%sCI = %6.4f, HI = %5.2f for reference dose of %3.1f Gy\n',voiPrint,...
qi(runVoi).(['CI_' StringReferenceDose 'Gy']),qi(runVoi).(['HI_' StringReferenceDose 'Gy']),referenceDose);
end
end
matRad_dispToConsole(voiPrint,param,'info','%s\n')
else
matRad_dispToConsole([num2str(cst{runVoi,1}) ' ' cst{runVoi,2} ' - No dose information.\n'],param,'info')
end
end
% assign VOI names which could be corrupted due to empty structures
listOfFields = fieldnames(qi);
for i = 1:size(cst,1)
indices = cst{i,4}{1};
doseInVoi = sort(doseCube(indices));
if isempty(doseInVoi)
for j = 1:numel(listOfFields)
qi(i).(listOfFields{j}) = NaN;
end
qi(i).name = cst{i,2};
end
end
end