-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathreconstruct_dual.m
522 lines (486 loc) · 19.4 KB
/
reconstruct_dual.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
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
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
function reconstruct_dual(cosine_sample,phi_deg1,psi_deg1,Chosen_Filename_ali1,Chosen_Filename_angles1,phi_deg2,psi_deg2,Chosen_Filename_ali2,Chosen_Filename_angles2,varargin)
%(C) Written by Shahar Seifer, Weizmann Institute of Science, 2022
%Load aligned mrc tomograms and reconstruct 3D by SIRT3D in Astra toolbox -
%1 is for a and 2 is for b tilt series in dual axis tomogram
%Requires ASTRA Toolbox, https://www.astra-toolbox.com/downloads/index.html#
%Requires @MRCImage library from MatTomo, PEET project: https://bio3d.colorado.edu/imod/matlab.html
%varagin includes: bin,nZ
%##### check parameters ####
if false %to test independently
cosine_sample=0;
phi_deg1=90;
psi_deg1=0;
phi_deg2=90;
psi_deg2=0;
Chosen_Filename_ali1='D:\results\AD68_1_pt12a.mrc';
Chosen_Filename_ali2='D:\results\AD68_1_pt12b.mrc';
Chosen_Filename_angles1='D:\results\AD68_1_pt12.rawtlt';
Chosen_Filename_angles2='D:\results\AD68_1_pt12.rawtlt';
varargin='';
end
if ~isempty(varargin)
bin=varargin{1};
if bin>8
disp(sprintf('Note that you have chosen bin=%g',bin));
end
else
bin=4; %binning (bin=4 is suitable for 8Gb GPU for two input of 60X2048X2048 images)
end
if length(varargin)==2
nZ=varargin{2};
if nZ<10
disp(sprintf('Note that you have chosen thickness of %g',nZ));
end
else
nZ=400; %thickness (in pixels after binning)
end
max_fiterr=15;
%###########################
gpu = gpuDevice();
reset(gpu);
if isempty(psi_deg1)
psi_deg1=0;
end
if isempty(phi_deg1)
phi_deg1=90;
end
if isempty(psi_deg2)
psi_deg2=0;
end
if isempty(phi_deg2)
phi_deg2=0;
end
rotateb=0;
if abs(phi_deg2-phi_deg1)<30
if phi_deg2>45
phi_deg2=phi_deg2-90;
rotateb=-90; % + is clockwise, so we rotate b image counterclockwise
else
%phi_deg2=phi_deg2+90;
%rotateb=+90; % + is clockwise
rotateb=-90; % ali from IMOD
phi_deg2=phi_deg2+rotateb;
end
end
phi1=phi_deg1*pi/180;
psi1=psi_deg1*pi/180;
phi2=phi_deg2*pi/180;
psi2=psi_deg2*pi/180;
rotation_xaxis1=(abs(cos(phi1))<0.7); %rough estimate for tapering margins only
rotation_xaxis2=(abs(cos(phi2))<0.7); %rough estimate for tapering margins only
flgLoadVolume=1; % If 1 - Load in the volume data (default: 1)
showHeader=1; % If 1 - Print out information as the header is loaded.
mRCImage=MRCImage;%Insentiate MRCImage in mRCImage
mRCImage = open(mRCImage, Chosen_Filename_ali1, flgLoadVolume, showHeader);
tilt = double(getVolume(mRCImage, [], [], []));
ntilts1 = getNZ(mRCImage);
nX = getNX(mRCImage);
nY = getNY(mRCImage);
sizeXangstrom=getCellX(mRCImage);
sizeYangstrom=sizeXangstrom;%getCellY(mRCImage) not used since it varies, but the actual is of the center angle 0
fileID = fopen(Chosen_Filename_angles1,'r');
formatSpec = '%g';
angles1 = fscanf(fileID,formatSpec);
fclose(fileID);
nX=round(nX/bin);
nY=round(nY/bin);
tilt1=zeros(nX,nY,ntilts1);
for n=1:ntilts1
tilt1(:,:,n)=imresize(tilt(:,:,n),[nX nY]);
end
mRCImage=MRCImage;%Insentiate MRCImage in mRCImage
mRCImage = open(mRCImage, Chosen_Filename_ali2, flgLoadVolume, showHeader);
tilt = double(getVolume(mRCImage, [], [], []));
ntilts2 = getNZ(mRCImage);
nX = getNX(mRCImage);
nY = getNY(mRCImage);
sizeXangstrom=getCellX(mRCImage);
sizeYangstrom=sizeXangstrom;%getCellY(mRCImage) not used since it varies, but the actual is of the center angle 0
fileID = fopen(Chosen_Filename_angles2,'r');
formatSpec = '%g';
angles2 = fscanf(fileID,formatSpec);
fclose(fileID);
nX=round(nX/bin);
nY=round(nY/bin);
tilt2=zeros(nX,nY,ntilts2);
for n=1:ntilts2
tilt2(:,:,n)=imresize(tilt(:,:,n),[nX nY]);
end
clear tilt;
%Correct rotation
for n=1:ntilts2
imag=tilt2(:,:,n);
imag=imrotate(imag,rotateb,'bilinear','crop'); %rotate + or -90 deg
tilt2(:,:,n)=imag;
end
%align b with respect to a, based on center slices
vect=1:length(angles1);
ind1=min(vect(abs(angles1)==min(abs(angles1))));
vect=1:length(angles2);
ind2=min(vect(abs(angles2)==min(abs(angles2))));
imag1=tilt1(:,:,ind1);
imag2=tilt2(:,:,ind2);
shift_limit=nX/4;
do_filt=1;
shiftB=r_mn_rect(imag2,imag1,shift_limit,do_filt);
imag2_toshow=imtranslate(imag2,shiftB); %to show center of tiltB aligned with center of tiltA
figure(1);
subplot(1,2,1);
balanced_imshow_in(tilt1(:,:,ind1));
subplot(1,2,2);
balanced_imshow_in(imag2_toshow);
pause(0.5);
%try
% Afiterr_all = readmatrix(Chosen_Filename_fiterr,'delimiter','\t');
% fiterr=Afiterr_all(length(Afiterr_all(:,1)),1:min([length(angles) ntilts]));
%catch
fiterr1=zeros(1,length(angles1));
fiterr2=zeros(1,length(angles2));
%remove slices that fiterr was not between 0 and 5
count=0;
for idx=1:min([length(angles1) ntilts1])
if fiterr1(idx)>=0 && fiterr1(idx)<=max_fiterr
count=count+1;
angles1(count)=angles1(idx);
tilt1(:,:,count)=tilt1(:,:,idx);
end
end
angles1=angles1(1:count);
count1=count;
count=0;
for idx=1:min([length(angles2) ntilts2])
if fiterr2(idx)>=0 && fiterr2(idx)<=max_fiterr
count=count+1;
angles2(count)=angles2(idx);
tilt2(:,:,count)=tilt2(:,:,idx);
end
end
angles2=angles2(1:count);
angles=zeros(1,length(angles1)+length(angles2));
angles(1:length(angles1))=angles1;
angles(1+length(angles1):length(angles1)+length(angles2))=angles2;
%Note it could crash if you lack a good GPU
det_row_count=nY;
det_col_count=nX;
sizeZangstrom=nZ*sizeXangstrom/nX;
vol_geom = astra_create_vol_geom(nY,nX,nZ);
rec_id = astra_mex_data3d('create', '-vol', vol_geom, 0);
proj_vectors=zeros(length(angles1)+length(angles2),12);
for idx=1:length(angles)
if idx<=count1
rotation_xaxis=rotation_xaxis1;
phi=phi1;
psi=psi1;
else
rotation_xaxis=rotation_xaxis2;
phi=phi2;
psi=psi2;
end
%transformation around axis (cos(psi)sin(phi),cos(psi)cos(phi),sin(psi))
rayX=(1-cos(angles(idx)*pi/180))*cos(psi)*sin(psi)*sin(-phi)+sin(angles(idx)*pi/180)*cos(psi)*cos(-phi);
rayY=(1-cos(angles(idx)*pi/180))*cos(psi)*sin(psi)*cos(-phi)-sin(angles(idx)*pi/180)*cos(psi)*sin(-phi);
rayZ=cos(angles(idx)*pi/180)+(1-cos(angles(idx)*pi/180))*(sin(psi))^2;
if idx<=count1
dX=0;
dY=0;
dZ=0;
else %center of detector (in 3D space) means where the center of detector (and center of rotation!) is within the output volume
dX=shiftB(2);
dY=shiftB(1);
dZ=0; %Add here relative Z shift between the experiments if needed
end
if cosine_sample
if rotation_xaxis
vX=0; %for row shift of one pixel in the detector actual:(0,1)
vY=cos(angles(idx)*pi/180)*cos(angles(idx)*pi/180);
vZ=-sin(angles(idx)*pi/180)*cos(angles(idx)*pi/180)*sign(phi);
else
uY=0; %for row shift of one pixel in the detector actual:(1,0)
uX=cos(angles(idx)*pi/180)*cos(angles(idx)*pi/180);
uZ=-sin(angles(idx)*pi/180)*cos(angles(idx)*pi/180);
end
else
if rotation_xaxis %phi=90deg or -90deg
vX=0; %for row shift of one pixel in the detector actual:(0,1)
vY=cos(angles(idx)*pi/180);
vZ=-sin(angles(idx)*pi/180)*sign(phi);
else %phi=0
uX=cos(angles(idx)*pi/180);
uY=0;
uZ=-sin(angles(idx)*pi/180);
end
end
if rotation_xaxis
uX=1; %for column shift of one pixel in the detector actual (1,0)
uY=0;
uZ=0;
else
vY=1;
vX=0;
vZ=0;
end
proj_vectors(idx,:)=[rayX rayY rayZ dX dY dZ uX uY uZ vX vY vZ];
end
mrg=0.1;
proj_geom = astra_create_proj_geom('parallel3d_vec', det_row_count+2*round(det_row_count*mrg), det_col_count+2*round(det_col_count*mrg), proj_vectors);
%
% Create a 3D parallel beam geometry specified by 3D vectors.
% det_row_count: number of detector rows in a single projection
% det_col_count: number of detector columns in a single projection
% vectors: a matrix containing the actual geometry. Each row corresponds
% to a single projection, and consists of:
% ( rayX, rayY, rayZ, dX, dY, dZ, uX, uY, uZ, vX, vY, vZ )
% ray: the ray direction
% d : the center of the detector
% u : the vector from detector pixel (0,0) to (1,0) -this is for x axis shift (correction in relation to manual, but consistent with order of u and v)
% v : the vector from detector pixel (0,0) to (0,1) -The order in the imag/mrc file is (column,row), so this is for primary y' axis
%
% proj_geom: MATLAB struct containing all information of the geometry
proj_data_mat=zeros(nX+2*round(mrg*nX),length(angles),nY+2*round(mrg*nY));
%compensate for acquisition in adaptive aspect ratio
for idx=1:length(angles)
if idx<=count1
rotation_xaxis=rotation_xaxis1;
%continue;
else
rotation_xaxis=rotation_xaxis2;
%continue;
end
numrows=nY;
numcols=nX;
if idx<=count1
imag=imresize(tilt1(:,:,idx),[numcols numrows]);
else
imag=imresize(tilt2(:,:,idx-count1),[numcols numrows]);
end
simag1=mean(imag(round(0.35*numcols):round(0.65*numcols),:),1);%average of every line (parallel to x axis, the rotation), so it is easy to detect margins from alignments
del_lines=simag1(2:numrows)-simag1(1:numrows-1);
del_lines_usual=median(abs(del_lines));
simag1c=mean(imag(:,round(0.35*numrows):round(0.65*numrows)),2);%average of every col
del_linesc=simag1c(2:numcols)-simag1c(1:numcols-1);
del_lines_usualc=median(abs(del_linesc));
vect=3:round(0.3*numrows);
if max(abs(del_lines(vect)))>10*del_lines_usual
margin_row1=min(vect(abs(del_lines(vect))==max(abs(del_lines(vect)))))+1;
temp1=double(imag(round(0.35*numcols):round(0.65*numcols),2:margin_row1-2));
temp2=double(imag(round(0.35*numcols):round(0.65*numcols),margin_row1:numrows));
if std(temp1(:))>0.25*std(temp2(:))
margin_row1=1;
else
margin_row1=margin_row1+1;
end
else
margin_row1=1;
end
vect=round(0.7*numrows):numrows-2;
if max(abs(del_lines(vect)))>10*del_lines_usual
margin_row2=max(vect(abs(del_lines(vect))==max(abs(del_lines(vect)))));
temp1=double(imag(round(0.35*numcols):round(0.65*numcols),margin_row2+2:numrows));
temp2=double(imag(round(0.35*numcols):round(0.65*numcols),margin_row1:margin_row2));
if std(temp1(:))>0.25*std(temp2(:))
margin_row2=numrows;
else
margin_row2=margin_row2-1;
end
else
margin_row2=numrows;
end
vectc=3:round(0.3*numcols);
if max(abs(del_linesc(vectc)))>10*del_lines_usualc
margin_col1=min(vectc(abs(del_linesc(vectc))==max(abs(del_linesc(vectc)))))+1;
temp1=double(imag(2:margin_col1-2,round(0.35*numrows):round(0.65*numrows)));
temp2=double(imag(margin_col1:numcols,round(0.35*numrows):round(0.65*numrows)));
if std(temp1(:))>0.25*std(temp2(:))
margin_col1=1;
else
margin_col1=margin_col1+1;
end
else
margin_col1=1;
end
vectc=round(0.7*numcols):numcols-2;
if max(abs(del_linesc(vectc)))>10*del_lines_usualc
margin_col2=max(vectc(abs(del_linesc(vectc))==max(abs(del_linesc(vectc)))));
temp1=double(imag(margin_col2+2:numcols,round(0.35*numrows):round(0.65*numrows)));
temp2=double(imag(margin_col1:margin_col2,round(0.35*numrows):round(0.65*numrows)));
if std(temp1(:))>0.25*std(temp2(:))
margin_col2=numcols;
else
margin_col2=margin_col2-1;
end
else
margin_col2=numcols;
end
%erode more from the edges
margin_col1=margin_col1+round(nX/40);
margin_col2=margin_col2-round(nX/40);
margin_row1=margin_row1+round(nY/40);
margin_row2=margin_row2-round(nY/40);
%normalize image intensity
croped_imag=imag(margin_col1:margin_col2,margin_row1:margin_row2);
croped_imag=5000*(croped_imag-imgaussfilt(croped_imag,round(numcols/40)));
imag=zeros(size(imag));
imag(margin_col1:margin_col2,margin_row1:margin_row2)=croped_imag;
margin_up=margin_row1;
margin_down=margin_row2;
margin_left=margin_col1;
margin_right=margin_col2;
imfull=zeros(nX+2*round(mrg*nX),nY+2*round(mrg*nY));
imfull(margin_left+round(mrg*nX):margin_right+round(mrg*nX),margin_up+round(mrg*nY):margin_down+round(mrg*nY))=imag(margin_left:margin_right,margin_up:margin_down);
imfull_temp=imfull;
%imfull(1+round(mrg*nX):nX+round(mrg*nX),1+round(mrg*nY):nY+round(mrg*nY))=imag;
%Fill in the margins by mirror image of the inside content
imfull(1:margin_left+round(mrg*nX)+1,:)=imfull_temp(2*margin_left+2*round(mrg*nX)+2:-1:margin_left+round(mrg*nX)+2,:);
imfull(margin_right+round(mrg*nX)-1:nX+2*round(mrg*nX),:)=imfull_temp(margin_right+round(mrg*nX)-2:-1:2*margin_right-nX-3,:);
imfull_temp=imfull;
imfull(:,1:margin_up+round(mrg*nY)+1)=imfull_temp(:,2*margin_up+2*round(mrg*nY)+2:-1:margin_up+round(mrg*nY)+2);
imfull(:,margin_down+round(mrg*nY)-1:nY+2*round(mrg*nY))=imfull_temp(:,margin_down+round(mrg*nY)-2:-1:2*margin_down-nY-3);
if idx<=count1
if ~cosine_sample && rotation_xaxis
margin_up=max(margin_row1,1+round(round(numrows/2)*(1-cos(angles(idx)*pi/180))));
margin_down=min(margin_row2,round(round(numrows/2)*(1+cos(angles(idx)*pi/180))));
end
if ~cosine_sample && ~rotation_xaxis
margin_left=max(margin_col1,1+round(round(numcols/2)*(1-cos(angles(idx)*pi/180))));
margin_right=min(margin_col2,round(round(numcols/2)*(1+cos(angles(idx)*pi/180))));
end
end
%Tapping unimportant regions
imfull_temp=imfull;
for n=1:margin_up+round(mrg*nY)
imfull(:,n)=imfull_temp(:,n)*double(n/(margin_up+round(mrg*nY))).^4;
end
for n=margin_down+round(mrg*nY):numrows+2*round(mrg*nY)
imfull(:,n)=imfull_temp(:,n)*double((numrows+2*round(mrg*nY)-n)/(numrows+round(mrg*nY)-margin_down)).^4;
end
imfull_temp=imfull;
for n=1:margin_left+round(mrg*nX)
imfull(n,:)=imfull_temp(n,:)*double(n/(margin_left+round(mrg*nX))).^4;
end
for n=margin_right+round(mrg*nX):numcols+2*round(mrg*nX)
imfull(n,:)=imfull_temp(n,:)*double((numcols+2*round(mrg*nX)-n)/(numcols+round(mrg*nX)-margin_right)).^4;
end
figure(3)
balanced_imshow(imfull);
pause(0.5);
%mask=zeros(size(imag));
%mask(:,margin_up:margin_down)=1;
proj_data_mat(:,idx,:)=permute(imfull,[1 3 2]); % order should be: column(=x), angle, rows=y
%mask_data_mat(:,idx,:)=permute(mask,[1 3 2]); % order should be: column(=x), angle, rows=y
end
proj_id = astra_mex_data3d('create', '-proj3d', proj_geom, proj_data_mat);
% Set up the parameters for a reconstruction algorithm using the GPU
cfg = astra_struct('SIRT3D_CUDA');
%cfg = astra_struct('CGLS3D_CUDA');
cfg.ReconstructionDataId = rec_id;
cfg.ProjectionDataId = proj_id;
%cfg.option.SinogramMaskId=mask_id; %only with sirt3d and helps only at the margins of the volume slices
% Create the algorithm object from the configuration structure
alg_id = astra_mex_algorithm('create', cfg);
% Run 50 iterations of the algorithm
astra_mex_algorithm('iterate', alg_id,150);
% Get the result
rec = astra_mex_data3d('get', rec_id);%maybe 'get_single'
%errorP=astra_mex_algorithm('get_res_norm', alg_id)/((nX)*(nY)*length(angles));
%Save to new MRC names rec_...
if contains(Chosen_Filename_ali1, '_jali.mrc')
newFilename=strrep(Chosen_Filename_ali1,'_jali.mrc','_dual_rec_SIRT.mrc');
elseif contains(Chosen_Filename_ali1, '_ali.mrc')
newFilename=strrep(Chosen_Filename_ali1,'_ali.mrc','_dual_rec_SIRT.mrc');
elseif contains(Chosen_Filename_ali1, '.mrc')
newFilename=strrep(Chosen_Filename_ali1,'.mrc','_dual_rec_SIRT.mrc');
else
newFilename=[Chosen_Filename_ali1 '_dual_rec_SIRT.mrc'];
end
newmRCImage = MRCImage;%Instentiate MRCImage object
newmRCImage.filename=newFilename;
newmRCImage = setVolume(newmRCImage, rec); %enter to newmRCImage, do statistics, and fill many details to the header
newmRCImage.header.cellDimensionX = sizeXangstrom;
newmRCImage.header.cellDimensionY = sizeYangstrom;
newmRCImage.header.cellDimensionZ = sizeZangstrom;
%MODE=2: 32-bit signed real
save(newmRCImage, newFilename);
close(newmRCImage);
%disp(sprintf('Saved to file: %s ',newFilename));
%fprintf('Difference between source projections and reconstructions: %d \n',errorP);
astra_mex_data3d('clear');%free GPU memory
end %of main function
%####################################################
function r_mn=r_mn_rect(Imagem,Imagen,shift_limit,do_filt)
if do_filt==1
Imagem=imgaussfilt(Imagem-imgaussfilt(Imagem,30),3);
Imagen=imgaussfilt(Imagen-imgaussfilt(Imagen,30),3);
end
tempx=floor(0.3*size(Imagem,1)); % x are the row number, y is the col number (as observed with balanced_imshow). The rows progress along the first ordinate in Imagem/n.
tempy=floor(0.3*size(Imagem,2));
tempux=size(Imagem,1)-tempx;%floor(0.85*size(Imagem,1));
tempuy=size(Imagem,2)-tempy;%floor(0.7*size(Imagem,2));
view_in=Imagem(tempx:tempux,tempy:tempuy);
correlationOutput = normxcorr2(view_in,Imagen);
[maxCorrValue, maxIndex] = max(abs(correlationOutput(:)));
[xpeak, ypeak] = ind2sub(size(correlationOutput),maxIndex(1));%find(correlationOutput==max(correlationOutput(:))); xpeak is the row number
yoffset = ypeak-tempuy;
xoffset = xpeak-tempux;
if abs(yoffset)>shift_limit || abs(xoffset)>shift_limit
correlationOutput = normxcorr2(imgaussfilt(view_in,10),imgaussfilt(Imagen,10));
[maxCorrValue, maxIndex] = max(abs(correlationOutput(:)));
[xpeak, ypeak] = ind2sub(size(correlationOutput),maxIndex(1));%find(correlationOutput==max(correlationOutput(:)));
yoffset = ypeak-tempuy;
xoffset = xpeak-tempux;
if abs(yoffset)>shift_limit || abs(xoffset)>shift_limit
r_mn=[0 0];
else
r_mn=[yoffset xoffset];
end
disp('Only rough shift estimate')
return;
end
%refine to subpixel
sample16=correlationOutput(xpeak-7:xpeak+8,ypeak-7:ypeak+8);
Intsample16=fftInterpolate(sample16,[512 512]);
[maxCorrValue2, maxIndex2] = max(abs(Intsample16(:)));
[xpeak2, ypeak2] = ind2sub(size(Intsample16),maxIndex2(1));%find(Intsample16==max(Intsample16(:)));
yoffset2=yoffset+(ypeak2-256+30)/32;
xoffset2=xoffset+(xpeak2-256+31)/32;
r_mn=[yoffset2 xoffset2];
end
function OK=balanced_imshow_in(img)
Nshades=1024;
mapvector=linspace(0,1,Nshades)';
cmap=zeros(Nshades,3);
for loop=1:3
cmap(:,loop)=mapvector;
end
try
showpic2=balance(img,Nshades);
OK=imshow(showpic2',cmap); %Here is the built in function to show images in Matlab
catch
OK=imshow(img);
end
function normpic2=balance(normpic,Nshades)
[BinValues,BinEdges]=histcounts(normpic,Nshades);
NumBins=length(BinValues);
sumH=sum(BinValues);
temp=0;
lowedge=BinEdges(NumBins);
for n=1:NumBins-1
temp=temp+BinValues(n);
if temp>0.005*sumH
lowedge=BinEdges(n);
break;
end
end
temp=0;
highedge=BinEdges(1);
for n2=NumBins:-1:2
temp=temp+BinValues(n2);
if temp>0.005*sumH
highedge=BinEdges(n2);
break;
end
end
normpic(normpic>highedge)=highedge; %remove white dots
normpic(normpic<lowedge)=lowedge; %remove black dots
normpic2=((double(normpic)-lowedge)*Nshades)/double(highedge-lowedge);
end
end