-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathprocessImages.m
More file actions
176 lines (150 loc) · 8.07 KB
/
processImages.m
File metadata and controls
176 lines (150 loc) · 8.07 KB
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
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
function [K2_fundamental,K2_total,K2_shot,...
K2_read,K2_quantized,K2_spatial,cc,cc_fit] ...
= processImages(imgs,mask,gain,dark_var,mean_img,mean_img_num,use_intensity_weight,use_lsq_fit,window_size,consider_spatial_heterogeneity)
%processImages Process image to get corrected K2.
% imgs = 3D double (y,x,time). Only provide images within ROI to be
% processed.
% gain = matrix of gain map. ADU/e-
% dark_var = dark image temporal variance. Should already have quantization
% noise variance subtracted.
% mean_img = 2D double (y,x)
% use_intensity_weight = Boolean. If true, weighs K2 values by their
% intensities.
% use_lsq_fit
% window_size = 1 or 2 element vector (y,x).
if numel(window_size) == 1
window_size_y = window_size;
window_size_x = window_size;
else
window_size_y = window_size(1);
window_size_x = window_size(2);
end
frame_num = size(imgs,3);
x_window_start = 1:window_size_x:size(imgs,2);
x_window_start(size(imgs,2)-x_window_start < window_size_x - 1) = []; % only if there is enough buffer
y_window_start = 1:window_size_y:size(imgs,1);
y_window_start(size(imgs,1)-y_window_start < window_size_y - 1) = [];
K2_total_window = nan(length(y_window_start),length(x_window_start),frame_num);
K2_nonlinear_window = K2_total_window;
K2_shot_window = K2_total_window; K2_read_window = K2_total_window; K2_quantization_window = K2_total_window;
K2_spatial_window = K2_total_window; cc_window = K2_total_window; cc_window_fit = K2_total_window;
cc_window_norm = K2_total_window;
good_window = false(length(y_window_start),length(x_window_start));
cc_mean = squeeze(mean(mean(imgs,1),2));
cc_mean_smooth = smooth(cc_mean); % the temporal pattern that is expected
% find whether there is enough variance expected
expected_var = mean(cc_mean)*gain;
quantization_threshold = 0.42;
% fitting function
options = optimoptions('lsqcurvefit','Algorithm','levenberg-marquardt','Display','off');
lb = [];
ub = [];
x0=[0,0];
fun = @(x,Ipattern_ts)(x(1).*Ipattern_ts+x(2));
% intensity weight
intensity_weight = mean_img./mean(mean_img(:));
% for each window
for ii=1:length(y_window_start)
for jj=1:length(x_window_start)
x_window = x_window_start(jj):x_window_start(jj)+window_size_x-1;
y_window = y_window_start(ii):y_window_start(ii)+window_size_y-1;
imgs_window = imgs(y_window,x_window,:);
mask_window = mask(y_window,x_window);
mask_window = mask_window(:); % extend to column
% var_diff_window_map = mean(var_diff(:,x_window),2);
good_window(ii,jj) = sum(mask_window)./numel(mask_window) >= 0.5;
if good_window(ii,jj)
imgs_window_masked = reshape(imgs_window,[],size(imgs_window,3)); % also extend to column
imgs_window_masked = imgs_window_masked(mask_window,:);
cc_window_masked = mean(imgs_window_masked,1);
if use_lsq_fit
x = lsqcurvefit(fun,x0,cc_mean_smooth(:),cc_window_masked',lb,ub,options);
cc_window_masked_fit = x(1).*cc_mean_smooth(:)+x(2); % the rescaled pattern for this patch
else
cc_window_masked_fit = cc_window_masked';
end
% get average gain for the window
gain_window = gain(y_window,x_window);
gain_window = mean(gain_window(:),"omitnan");
dark_var_window = mean(mean(dark_var(y_window,x_window)));
% spatial heterogeneity
mean_img_window = mean_img(y_window,x_window);
var_spatial = var(mean_img_window(:));
mean_spatial = mean(mean_img_window(:));
var_spatial = var_spatial - gain_window*mean_spatial/mean_img_num; % remove other noise contributions from mean img
if ~consider_spatial_heterogeneity
var_spatial = 0;
end
% get var_diff
% var_diff_window(ii,jj) = interp1(x_interp,var_diff_window_map,mean(cc_window_masked_fit(:)));
K2_total_window(ii,jj,:) = var(imgs_window_masked,0,1)'./(cc_window_masked_fit.^2);
% K2_nonlinear_window(ii,jj,:) = repmat(var_diff_window(ii,jj),size(imgs_window_masked,2),1)./(cc_window_masked_fit.^2);
K2_shot_window(ii,jj,:) = shotNoiseK2(cc_window_masked_fit,gain_window);
K2_read_window(ii,jj,:) = dark_var_window./(cc_window_masked_fit.^2);
K2_quantization_window(ii,jj,:) = 1./12./(cc_window_masked_fit.^2);
K2_spatial_window(ii,jj,:) = var_spatial./(mean_spatial^2)*ones(size(cc_window_masked_fit));
cc_window(ii,jj,:) = cc_window_masked;
cc_window_fit(ii,jj,:) = cc_window_masked_fit;
cc_window_norm(ii,jj,:) = mean(mean(intensity_weight(y_window,x_window),1),2);
if use_intensity_weight
K2_total_window(ii,jj,:) = ...
squeeze(mean(mean(K2_total_window(ii,jj,:).*cc_window_norm(ii,jj,:).^2,1),2));
% K2_nonlinear_window(ii,jj,:) = ...
% squeeze(mean(mean(K2_nonlinear_window(ii,jj,:).*cc_window_norm(ii,jj,:).^2,1),2));
K2_shot_window(ii,jj,:) = ...
squeeze(mean(mean(K2_shot_window(ii,jj,:).*cc_window_norm(ii,jj,:).^2,1),2));
K2_read_window(ii,jj,:) = ...
squeeze(mean(mean(K2_read_window(ii,jj,:).*cc_window_norm(ii,jj,:).^2,1),2));
K2_quantization_window(ii,jj,:) = ...
squeeze(mean(mean(K2_quantization_window(ii,jj,:).*cc_window_norm(ii,jj,:).^2,1),2));
% K2_spatial_window(ii,jj,:) = ...
% squeeze(mean(mean(K2_spatial_window(ii,jj,:).*cc_window_norm(ii,jj,:).^2,1),2));
end
else
K2_total_window(ii,jj,:) = nan;
% K2_nonlinear_window(ii,jj,:) = nan;
K2_shot_window(ii,jj,:) = nan;
K2_read_window(ii,jj,:) = nan;
K2_quantization_window(ii,jj,:) = nan;
K2_spatial_window(ii,jj,:) = nan;
cc_window(ii,jj,:) = nan;
cc_window_fit(ii,jj,:) = nan;
cc_window_norm(ii,jj,:) = nan;
end
end
end
K2_fundamental_window = K2_total_window - K2_shot_window - K2_read_window - K2_quantization_window - K2_spatial_window;
% K2_fundamental_window = K2_total_window - K2_shot_window - K2_read_window;
% only select good roi
K2_total_window = reshape(K2_total_window,[],size(K2_total_window,3));
K2_total_window = K2_total_window(good_window(:),:);
% K2_nonlinear_window = reshape(K2_nonlinear_window,[],size(K2_nonlinear_window,3));
% K2_nonlinear_window = K2_nonlinear_window(good_window(:),:);
K2_shot_window = reshape(K2_shot_window,[],size(K2_shot_window,3));
K2_shot_window = K2_shot_window(good_window(:),:);
K2_read_window = reshape(K2_read_window,[],size(K2_read_window,3));
K2_read_window = K2_read_window(good_window(:),:);
K2_quantization_window = reshape(K2_quantization_window,[],size(K2_quantization_window,3));
K2_quantization_window = K2_quantization_window(good_window(:),:);
K2_spatial_window = reshape(K2_spatial_window,[],size(K2_spatial_window,3));
K2_spatial_window = K2_spatial_window(good_window(:),:);
K2_fundamental_window = reshape(K2_fundamental_window,[],size(K2_fundamental_window,3));
K2_fundamental_window = K2_fundamental_window(good_window(:),:);
cc_window = reshape(cc_window,[],size(cc_window,3));
cc_window = cc_window(good_window(:),:);
cc_window_fit = reshape(cc_window_fit,[],size(cc_window_fit,3));
cc_window_fit = cc_window_fit(good_window(:),:);
cc_window_norm = reshape(cc_window_norm,[],size(cc_window_norm,3));
cc_window_norm = cc_window_norm(good_window(:),:);
% get mean
K2_total = mean(K2_total_window,1,"omitnan")';
% K2_nonlinear = mean(K2_nonlinear_window,1)';
K2_shot = mean(K2_shot_window,1,"omitnan")';
K2_read = mean(K2_read_window,1,"omitnan")';
K2_quantized = mean(K2_quantization_window,1,"omitnan")';
K2_spatial = mean(K2_spatial_window,1,"omitnan")';
K2_fundamental = mean(K2_fundamental_window,1,"omitnan")';
cc = mean(cc_window,1,"omitnan")';
cc_fit = mean(cc_window_fit,1,"omitnan")';
cc_ROI_norm = mean(cc_window_norm,1,"omitnan")';
end