source: trunk/src/geometry_calib.m @ 696

Last change on this file since 696 was 696, checked in by sommeria, 10 years ago

bugs corrected

File size: 54.9 KB
Line 
1%'geometry_calib': associated to the GUI geometry_calib to perform geometric calibration from a set of reference points
2%------------------------------------------------------------------------
3% function hgeometry_calib = geometry_calib(inputfile,pos)
4%
5%OUTPUT:
6% hgeometry_calib=current handles of the GUI geometry_calib.fig
7%
8%INPUT:
9% inputfile: (optional) name of an xml file containing coordinates of reference points
10% pos: (optional) 4 element vector setting the 'Position' of the GUI
11%
12%A%AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
13%  Copyright Joel Sommeria, 2008, LEGI / CNRS-UJF-INPG, sommeria@coriolis-legi.org.
14%AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
15%     This file is part of the toolbox UVMAT.
16%
17%     UVMAT is free software; you can redistribute it and/or modify
18%     it under the terms of the GNU General Public License as published by
19%     the Free Software Foundation; either version 2 of the License, or
20%     (at your option) any later version.
21%
22%     UVMAT is distributed in the hope that it will be useful,
23%     but WITHOUT ANY WARRANTY; without even the implied warranty of
24%     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
25%     GNU General Public License (file UVMAT/COPYING.txt) for more details.
26%AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
27
28function varargout = geometry_calib(varargin)
29% GEOMETRY_CALIB M-file for geometry_calib.fig
30%      GEOMETRY_CALIB, by itself, creates a MenuCoord GEOMETRY_CALIB or raises the existing
31%      singleton*.
32%
33%      H = GEOMETRY_CALIB returns the handle to a MenuCoord GEOMETRY_CALIB or the handle to
34%      the existing singleton*.
35%
36%      GEOMETRY_CALIB('CALLBACK',hObject,eventData,handles,...) calls the local
37%      function named CALLBACK in GEOMETRY_CALIB.M with the given input arguments.
38%
39%      GEOMETRY_CALIB('Property','Value',...) creates a MenuCoord GEOMETRY_CALIB or raises the
40%      existing singleton*.  Starting from the left, property value pairs are
41%      applied to the GUI before geometry_calib_OpeningFunction gets called.  An
42%      unrecognized property name or invalid value makes property application
43%      stop.  All inputs are passed to geometry_calib_OpeningFcn via varargin.
44%
45%      *See GUI Options on GUIDE's Tools menu.  Choose "GUI allows only one
46%      instance to run (singleton)".
47%
48% See also: GUIDE, GUIDATA, GUIHANDLES
49
50% Edit the above text to modify the response to help geometry_calib
51
52% Last Modified by GUIDE v2.5 29-Oct-2013 06:46:10
53
54% Begin initialization code - DO NOT edit
55gui_Singleton = 1;
56gui_State = struct('gui_Name',       mfilename, ...
57                   'gui_Singleton',  gui_Singleton, ...
58                   'gui_OpeningFcn', @geometry_calib_OpeningFcn, ...
59                   'gui_OutputFcn',  @geometry_calib_OutputFcn, ...
60                   'gui_LayoutFcn',  [] , ...
61                   'gui_Callback',   []);
62if nargin
63   [pp,ff]=fileparts(which(varargin{1})); % name of the input file
64   if strcmp(ff,mfilename)% if we are activating a sub-function of geometry_calib
65   % ~isempty(regexp(varargin{1},'_Callback','once'))
66    gui_State.gui_Callback = str2func(varargin{1});
67   end
68end
69
70if nargout
71    [varargout{1:nargout}] = gui_mainfcn(gui_State, varargin{:});
72else
73    gui_mainfcn(gui_State, varargin{:});
74end
75% End initialization code - DO NOT edit
76
77
78% --- Executes just before geometry_calib is made visible.
79%INPUT:
80%handles: handles of the geometry_calib interface elements
81% PlotHandles: set of handles of the elements contolling the plotting
82% parameters on the uvmat interface (obtained by 'get_plot_handle.m')
83%------------------------------------------------------------------------
84function geometry_calib_OpeningFcn(hObject, eventdata, handles,inputfile)
85%------------------------------------------------------------------------
86% Choose default command line output for geometry_calib
87
88handles.output = hObject;
89
90% Update handles structure
91guidata(hObject, handles);
92set(hObject,'DeleteFcn',{@closefcn})%
93%set(hObject,'WindowButtonDownFcn',{'mouse_alt_gui',handles}) % allows mouse action with right button (zoom for uicontrol display)
94
95%% position
96set(0,'Unit','pixels')
97ScreenSize=get(0,'ScreenSize');% get the size of the screen, to put the fig on the upper right
98Left=ScreenSize(3)- 460; %right edge close to the right, with margin=40 (GUI width=420 px)
99if ScreenSize(4)>920
100    Height=840;%default height of the GUI
101    Bottom=ScreenSize(4)-Height-40; %put fig at top right
102else
103    Height=ScreenSize(4)-80;
104    Bottom=40; % GUI lies o the screen bottom (with margin =40)
105end
106set(handles.calib_type,'Position',[1 Height-40 194 30])%  rank 1
107set(handles.APPLY,'Position',[197 Height-40 110 30])%  rank 1
108set(handles.REPLICATE,'Position',[309 Height-40 110 30])%  rank 1
109set(handles.Intrinsic,'Position',[1 Height-40-2-92 418 92])%  rank 2
110set(handles.Extrinsic,'Position',[1 Height-40-4-92-75 418 75])%  rank 3
111set(handles.PointLists,'Position',[1 Height-40-6-92-75-117 418 117]) %  rank 4
112set(handles.CheckEnableMouse,'Position',[3 Height-40-8-92-75-117-30 203 30])%  rank 5
113set(handles.PLOT,'Position',[3 Height-394 120 30])%  rank 6
114set(handles.Copy,'Position',[151 Height-394 120 30])%  rank 6
115set(handles.CLEAR_PTS,'Position',[297 Height-394 120 30])%  rank 6
116set(handles.ClearLine,'Position',[297 Height-364 120 30])%  rank 6
117set(handles.phys_title,'Position',[38 Height-426 125 20])%  rank 7
118set(handles.CoordUnit,'Position',[151 Height-426 120 30])%  rank 7
119set(handles.px_title,'Position',[272 Height-426 125 20])%  rank 7
120set(handles.ListCoord,'Position',[1 20 418 Height-446])% rank 8
121set(handles.geometry_calib,'Position',[Left Bottom 420 Height])
122
123%set menu of calibration options
124set(handles.calib_type,'String',{'rescale';'linear';'3D_linear';'3D_quadr';'3D_extrinsic'})
125if exist('inputfile','var')&& ~isempty(inputfile)
126    struct.XmlInputFile=inputfile;
127    [RootPath,SubDir,RootFile,tild,tild,tild,tild,FileExt]=fileparts_uvmat(inputfile);
128    if ~strcmp(FileExt,'.xml')
129        inputfile=fullfile(RootPath,[SubDir '.xml']);%xml file corresponding to the input file
130        if ~exist(inputfile,'file')% case of civ files , removes the extension for subdir
131            inputfile=fullfile(RootPath,[regexprep(SubDir,'\..+$','') '.xml']);
132            if ~exist(inputfile,'file')
133                inputfile=[fullfile(RootPath,SubDir,RootFile) '.xml'];%old convention
134                if ~exist(inputfile,'file')
135                    inputfile='';
136                end
137            end
138        end
139    end
140    set(handles.ListCoord,'Data',[])
141    if exist(inputfile,'file')
142        Heading=loadfile(handles,inputfile);% load data from the xml file
143        if isfield(Heading,'Campaign')&& ischar(Heading.Campaign)
144            struct.Campaign=Heading.Campaign;
145        end
146    end   
147    set(hObject,'UserData',struct)
148end
149
150%------------------------------------------------------------------------
151% --- Outputs from this function are returned to the command line.
152function varargout = geometry_calib_OutputFcn(~, eventdata, handles)
153%------------------------------------------------------------------------
154% Get default command line output from handles structure
155varargout{1} = handles.output;
156varargout{2}=handles;
157%
158%------------------------------------------------------------------------
159% executed when closing: set the parent interface button to value 0
160function closefcn(gcbo,eventdata)
161%------------------------------------------------------------------------
162huvmat=findobj(allchild(0),'Name','uvmat');
163if ~isempty(huvmat)
164    handles=guidata(huvmat);
165    set(handles.MenuCalib,'Checked','off')
166    hobject=findobj(handles.PlotAxes,'tag','calib_points');
167    if ~isempty(hobject)
168        delete(hobject)
169    end
170    hobject=findobj(handles.PlotAxes,'tag','calib_marker');
171    if ~isempty(hobject)
172        delete(hobject)
173    end   
174end
175
176%------------------------------------------------------------------------
177% --- Executes on button press APPLY (used to launch the calibration).
178function APPLY_Callback(hObject, eventdata, handles)
179%------------------------------------------------------------------------
180%% look for the GUI uvmat and check for an image as input
181set(handles.APPLY,'BackgroundColor',[1 1 0])
182huvmat=findobj(allchild(0),'Name','uvmat');
183hhuvmat=guidata(huvmat);%handles of elements in the GUI uvmat
184
185RootPath='';
186if ~isempty(hhuvmat.RootPath)&& ~isempty(hhuvmat.RootFile)
187    RootPath=get(hhuvmat.RootPath,'String');
188    SubDirBase=regexprep(get(hhuvmat.SubDir,'String'),'\..+$','');
189    outputfile=[fullfile(RootPath,SubDirBase) '.xml'];%xml file associated with the currently displayed image
190else
191    question={'save the calibration data and point coordinates in'};
192    def={fullfile(RootPath,'ObjectCalib.xml')};
193    options.Resize='on';
194    answer=inputdlg(question,'',1,def,options);
195    outputfile=answer{1};
196end
197[GeometryCalib,index]=calibrate(handles,hhuvmat);% apply calibration
198
199if isempty(GeometryCalib) % if calibration cancelled
200    set(handles.APPLY,'BackgroundColor',[1 0 1])
201else   % if calibration confirmed
202   
203    %% copy the xml file from the old location if appropriate, then update with the calibration parameters
204    if ~exist(outputfile,'file') && ~isempty(SubDirBase)
205        oldxml=[fullfile(RootPath,SubDirBase,get(hhuvmat.RootFile,'String')) '.xml'];
206        if exist(oldxml,'file')
207            [success,message]=copyfile(oldxml,outputfile);%copy the old xml file to a new one with the new convention
208        end
209    end
210    errormsg=update_imadoc(GeometryCalib,outputfile,'GeometryCalib');% introduce the calibration data in the xml file
211    if ~strcmp(errormsg,'')
212        msgbox_uvmat('ERROR',errormsg);
213    end
214   
215    %% display image with new calibration in the currently opened uvmat interface
216    hhh=findobj(hhuvmat.PlotAxes,'Tag','calib_marker');% delete calib points and markers
217    if ~isempty(hhh)
218        delete(hhh);
219    end
220    hhh=findobj(hhuvmat.PlotAxes,'Tag','calib_points');
221    if ~isempty(hhh)
222        delete(hhh);
223    end
224    set(hhuvmat.CheckFixLimits,'Value',0)% put FixedLimits option to 'off'
225    set(hhuvmat.CheckFixLimits,'BackgroundColor',[0.7 0.7 0.7])
226    UserData=get(handles.geometry_calib,'UserData');
227    UserData.XmlInputFile=outputfile;%save the current xml file name
228    set(handles.geometry_calib,'UserData',UserData)
229    uvmat('RootPath_Callback',hObject,eventdata,hhuvmat); %file input with xml reading  in uvmat, show the image in phys coordinates
230    PLOT_Callback(hObject, eventdata, handles)
231    Data=get(handles.ListCoord,'Data');
232    Data(:,6)=zeros(size(Data,1),1);
233    Data(index,6)=-1;% indicate in the list the point with max deviation (possible mistake)
234    set(handles.ListCoord,'Data',Data)% indicate in the list the point with max deviation (possible mistake)
235    figure(handles.geometry_calib)
236    set(handles.APPLY,'BackgroundColor',[1 0 0])
237end
238
239%------------------------------------------------------------------------
240% --- Executes on button press in REPLICATE
241function REPLICATE_Callback(hObject, eventdata, handles)
242%------------------------------------------------------------------------
243
244%% look for the GUI uvmat and check for an image as input
245huvmat=findobj(allchild(0),'Name','uvmat');
246hhuvmat=guidata(huvmat);%handles of elements in the GUI uvmat
247GeometryCalib=calibrate(handles,hhuvmat);% apply calibration
248
249%% open the GUI browse_data
250CalibData=get(handles.geometry_calib,'UserData');%read the calibration image source on the interface userdata
251if isfield(CalibData,'XmlInputFile')
252    InputDir=fileparts(fileparts(CalibData.XmlInputFile));
253end
254answer=msgbox_uvmat('INPUT_TXT','Campaign to calibrate?',InputDir);
255if strcmp(answer,'Cancel')
256    return
257end
258OutPut=browse_data(answer);
259nbcalib=0;
260for ilist=1:numel(OutPut.Experiment)
261    SubDirBase=regexprep(OutPut.Device{1},'\..+$','');
262    XmlName=fullfile(OutPut.Campaign,OutPut.Experiment{ilist},[SubDirBase '.xml']);
263    % copy the xml file from the old location if appropriate, then update with the calibration parameters
264    if ~exist(XmlName,'file') && ~isempty(SubDirBase)
265        oldxml=fullfile(OutPut.Campaign,OutPut.Experiment{ilist},SubDirBase,[get(hhuvmat.RootFile,'String') '.xml']);
266        if exist(oldxml,'file')
267            [success,message]=copyfile(oldxml,XmlName);%copy the old xml file to a new one with the new convention
268        end
269    end
270    errormsg=update_imadoc(GeometryCalib,XmlName,'GeometryCalib');% introduce the calibration data in the xml file
271    if ~strcmp(errormsg,'')
272        msgbox_uvmat('ERROR',errormsg);
273    else
274        display([XmlName ' updated with calibration parameters'])
275        nbcalib=nbcalib+1;
276    end
277end
278msgbox_uvmat('CONFIMATION',[SubDirBase ' calibrated for ' num2str(nbcalib) ' experiments']);
279
280%------------------------------------------------------------------------
281% --- activate calibration and store parameters in ouputfile .
282function [GeometryCalib,index]=calibrate(handles,hhuvmat)
283%------------------------------------------------------------------------
284%% read the current calibration points
285Coord=get(handles.ListCoord,'Data');
286Coord(:,6)=[];
287% apply the calibration, whose type is selected in  handles.calib_type
288if ~isempty(Coord)
289    calib_cell=get(handles.calib_type,'String');
290    val=get(handles.calib_type,'Value');
291    GeometryCalib=feval(['calib_' calib_cell{val}],Coord,handles);
292else
293    msgbox_uvmat('ERROR','No calibration points, abort')
294    return
295end
296Z_plane=[];
297if ~isempty(Coord)
298    %check error
299    X=Coord(:,1);
300    Y=Coord(:,2);
301    Z=Coord(:,3);
302    x_ima=Coord(:,4);
303    y_ima=Coord(:,5);
304    [Xpoints,Ypoints]=px_XYZ(GeometryCalib,X,Y,Z);
305    GeometryCalib.ErrorRms(1)=sqrt(mean((Xpoints-x_ima).*(Xpoints-x_ima)));
306    [GeometryCalib.ErrorMax(1),index(1)]=max(abs(Xpoints-x_ima));
307    GeometryCalib.ErrorRms(2)=sqrt(mean((Ypoints-y_ima).*(Ypoints-y_ima)));
308    [GeometryCalib.ErrorMax(2),index(2)]=max(abs(Ypoints-y_ima));
309    [tild,ind_dim]=max(GeometryCalib.ErrorMax);
310    index=index(ind_dim);
311    %set the Z position of the reference plane used for calibration
312    if isequal(max(Z),min(Z))%Z constant
313        Z_plane=Z(1);
314        GeometryCalib.NbSlice=1;
315        GeometryCalib.SliceCoord=[0 0 Z_plane];
316    end
317end
318%set the coordinate unit
319unitlist=get(handles.CoordUnit,'String');
320unit=unitlist{get(handles.CoordUnit,'value')};
321GeometryCalib.CoordUnit=unit;
322%record the points
323GeometryCalib.SourceCalib.PointCoord=Coord;
324display_intrinsic(GeometryCalib,handles)%display calibration intrinsic parameters
325
326% Display extrinsinc parameters (rotation and translation of camera with  respect to the phys coordiantes)
327set(handles.Tx,'String',num2str(GeometryCalib.Tx_Ty_Tz(1),4))
328set(handles.Ty,'String',num2str(GeometryCalib.Tx_Ty_Tz(2),4))
329set(handles.Tz,'String',num2str(GeometryCalib.Tx_Ty_Tz(3),4))
330set(handles.Phi,'String',num2str(GeometryCalib.omc(1),4))
331set(handles.Theta,'String',num2str(GeometryCalib.omc(2),4))
332set(handles.Psi,'String',num2str(GeometryCalib.omc(3),4))
333
334%% store the calibration data, by default in the xml file of the currently displayed image
335UvData=get(hhuvmat.uvmat,'UserData');
336NbSlice_j=1;%default
337ZStart=Z_plane;
338ZEnd=Z_plane;
339volume_scan='n';
340if isfield(UvData,'XmlData')
341    if isfield(UvData.XmlData,'TranslationMotor')
342        NbSlice_j=UvData.XmlData.TranslationMotor.Nbslice;
343        ZStart=UvData.XmlData.TranslationMotor.ZStart/10;
344        ZEnd=UvData.XmlData.TranslationMotor.ZEnd/10;
345        volume_scan='y';
346    end
347end
348
349answer=msgbox_uvmat('INPUT_Y-N',{'store calibration data';...
350    ['Error rms (along x,y)=' num2str(GeometryCalib.ErrorRms) ' pixels'];...
351    ['Error max (along x,y)=' num2str(GeometryCalib.ErrorMax) ' pixels']});
352
353%% get plane position(s)
354if ~strcmp(answer,'Yes')
355    GeometryCalib=[];
356    index=1;
357    return
358end
359if strcmp(calib_cell{val}(1:2),'3D')%set the plane position for 3D (projection) calibration
360    input_key={'Z (first position)','Z (last position)','Z (water surface)', 'refractive index','NbSlice','volume scan (y/n)','tilt angle y axis','tilt angle x axis'};
361    input_val=[{num2str(ZEnd)} {num2str(ZStart)} {num2str(ZStart)} {'1.333'} num2str(NbSlice_j) {volume_scan} {'0'} {'0'}];
362    answer=inputdlg(input_key,'slice position(s)',ones(1,8), input_val,'on');
363    GeometryCalib.NbSlice=str2double(answer{5});
364    GeometryCalib.VolumeScan=answer{6};
365    if isempty(answer)
366        Z_plane=0; %default
367    else
368        Z_plane=linspace(str2double(answer{1}),str2double(answer{2}),GeometryCalib.NbSlice);
369    end
370    GeometryCalib.SliceCoord=Z_plane'*[0 0 1];
371    GeometryCalib.SliceAngle(:,3)=0;
372    GeometryCalib.SliceAngle(:,2)=str2double(answer{7})*ones(GeometryCalib.NbSlice,1);%rotation around y axis (to generalise)
373    GeometryCalib.SliceAngle(:,1)=str2double(answer{8})*ones(GeometryCalib.NbSlice,1);%rotation around x axis (to generalise)
374    GeometryCalib.InterfaceCoord=[0 0 str2double(answer{3})];
375    GeometryCalib.RefractionIndex=str2double(answer{4});
376end
377
378
379
380%------------------------------------------------------------------------
381% determine the parameters for a calibration by an affine function (rescaling and offset, no rotation)
382function GeometryCalib=calib_rescale(Coord,handles)
383%------------------------------------------------------------------------
384X=Coord(:,1);
385Y=Coord(:,2);% Z not used
386x_ima=Coord(:,4);
387y_ima=Coord(:,5);
388[px]=polyfit(X,x_ima,1);
389[py]=polyfit(Y,y_ima,1);
390% T_x=px(2);
391% T_y=py(2);
392GeometryCalib.CalibrationType='rescale';
393GeometryCalib.fx_fy=[px(1) py(1)];%.fx_fy corresponds to pxcm along x and y
394GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
395GeometryCalib.Tx_Ty_Tz=[px(2)/px(1) py(2)/py(1) 1];
396GeometryCalib.omc=[0 0 0];
397
398%------------------------------------------------------------------------
399% determine the parameters for a calibration by a linear transform matrix (rescale and rotation)
400
401
402function GeometryCalib=calib_linear(Coord,handles)
403%------------------------------------------------------------------------
404X=Coord(:,1);
405Y=Coord(:,2);% Z not used
406x_ima=Coord(:,4);
407y_ima=Coord(:,5);
408XY_mat=[ones(size(X)) X Y];
409a_X1=XY_mat\x_ima; %transformation matrix for X
410a_Y1=XY_mat\y_ima;%transformation matrix for X
411R=[a_X1(2),a_X1(3);a_Y1(2),a_Y1(3)];
412epsilon=sign(det(R));
413norm=abs(det(R));
414GeometryCalib.CalibrationType='linear';
415if (a_X1(2)/a_Y1(3))>0
416    GeometryCalib.fx_fy(1)=sqrt((a_X1(2)/a_Y1(3))*norm);
417else
418    GeometryCalib.fx_fy(1)=-sqrt(-(a_X1(2)/a_Y1(3))*norm);
419end
420GeometryCalib.fx_fy(2)=(a_Y1(3)/a_X1(2))*GeometryCalib.fx_fy(1);
421GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
422GeometryCalib.Tx_Ty_Tz=[a_X1(1)/GeometryCalib.fx_fy(1) a_Y1(1)/GeometryCalib.fx_fy(2) 1];
423R(1,:)=R(1,:)/GeometryCalib.fx_fy(1);
424R(2,:)=R(2,:)/GeometryCalib.fx_fy(2);
425R=[R;[0 0]];
426GeometryCalib.R=[R [0;0;-epsilon]];
427GeometryCalib.omc=(180/pi)*[acos(GeometryCalib.R(1,1)) 0 0];
428
429%------------------------------------------------------------------------
430% determine the tsai parameters for a view normal to the grid plane
431% NOT USED
432function GeometryCalib=calib_normal(Coord,handles)
433%------------------------------------------------------------------------
434Calib.f1=str2num(get(handles.fx,'String'));
435Calib.f2=str2num(get(handles.fy,'String'));
436Calib.k=str2num(get(handles.kc,'String'));
437Calib.Cx=str2num(get(handles.Cx,'String'));
438Calib.Cy=str2num(get(handles.Cy,'String'));
439%default
440if isempty(Calib.f1)
441    Calib.f1=25/0.012;
442end
443if isempty(Calib.f2)
444    Calib.f2=25/0.012;
445end
446if isempty(Calib.k)
447    Calib.k=0;
448end
449if isempty(Calib.Cx)||isempty(Calib.Cy)
450    huvmat=findobj(allchild(0),'Tag','uvmat');
451    hhuvmat=guidata(huvmat);
452    Calib.Cx=str2num(get(hhuvmat.num_Npx,'String'))/2;
453    Calib.Cx=str2num(get(hhuvmat.num_Npy,'String'))/2;
454end   
455%tsai parameters
456Calib.dpx=0.012;%arbitrary
457Calib.dpy=0.012;
458Calib.sx=Calib.f1*Calib.dpx/(Calib.f2*Calib.dpy);
459Calib.f=Calib.f2*Calib.dpy;
460Calib.kappa1=Calib.k/(Calib.f*Calib.f);
461
462%initial guess
463X=Coord(:,1);
464Y=Coord(:,2);
465Zmean=mean(Coord(:,3));
466x_ima=Coord(:,4)-Calib.Cx;
467y_ima=Coord(:,5)-Calib.Cy;
468XY_mat=[ones(size(X)) X Y];
469a_X1=XY_mat\x_ima; %transformation matrix for X
470a_Y1=XY_mat\y_ima;%transformation matrix for Y
471R=[a_X1(2),a_X1(3),0;a_Y1(2),a_Y1(3),0;0,0,-1];% rotation+ z axis reversal (upward)
472norm=sqrt(det(-R));
473calib_param(1)=0;% quadratic distortion
474calib_param(2)=a_X1(1);
475calib_param(3)=a_Y1(1);
476calib_param(4)=Calib.f/(norm*Calib.dpx)-R(3,3)*Zmean;
477calib_param(5)=angle(a_X1(2)+1i*a_X1(3));
478display(['initial guess=' num2str(calib_param)])
479
480%optimise the parameters: minimisation of error
481calib_param = fminsearch(@(calib_param) error_calib(calib_param,Calib,Coord),calib_param);
482
483GeometryCalib.CalibrationType='tsai_normal';
484GeometryCalib.focal=Calib.f;
485GeometryCalib.dpx_dpy=[Calib.dpx Calib.dpy];
486GeometryCalib.Cx_Cy=[Calib.Cx Calib.Cy];
487GeometryCalib.sx=Calib.sx;
488GeometryCalib.kappa1=calib_param(1);
489GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
490GeometryCalib.Tx_Ty_Tz=[calib_param(2) calib_param(3) calib_param(4)];
491alpha=calib_param(5);
492GeometryCalib.R=[cos(alpha) sin(alpha) 0;-sin(alpha) cos(alpha) 0;0 0 -1];
493
494%------------------------------------------------------------------------
495function GeometryCalib=calib_3D_linear(Coord,handles)
496%------------------------------------------------------------------
497path_uvmat=which('uvmat');% check the path detected for source file uvmat
498path_UVMAT=fileparts(path_uvmat); %path to UVMAT
499huvmat=findobj(allchild(0),'Tag','uvmat');
500hhuvmat=guidata(huvmat);
501coord_files=get(handles.ListCoordFiles,'String');
502if ischar(coord_files)
503    coord_files={coord_files};
504end
505if isempty(coord_files{1}) || isequal(coord_files,{''})
506    coord_files={};
507end
508%retrieve the calibration points stored in the files listed in the popup list ListCoordFiles
509x_1=Coord(:,4:5)';%px coordinates of the ref points
510nx=str2num(get(hhuvmat.num_Npx,'String'));
511ny=str2num(get(hhuvmat.num_Npy,'String'));
512x_1(2,:)=ny-x_1(2,:);%reverse the y image coordinates
513X_1=Coord(:,1:3)';%phys coordinates of the ref points
514n_ima=numel(coord_files)+1;
515if ~isempty(coord_files)
516    msgbox_uvmat('CONFIRMATION',['The xy coordinates of the calibration points in ' num2str(n_ima) ' planes will be used'])
517    for ifile=1:numel(coord_files)
518    t=xmltree(coord_files{ifile});
519    s=convert(t);%convert to matlab structure
520        if isfield(s,'GeometryCalib')
521            if isfield(s.GeometryCalib,'SourceCalib')
522                if isfield(s.GeometryCalib.SourceCalib,'PointCoord')
523                PointCoord=s.GeometryCalib.SourceCalib.PointCoord;
524                Coord_file=zeros(length(PointCoord),5);%default
525                for i=1:length(PointCoord)
526                    line=str2num(PointCoord{i});
527                    Coord_file(i,4:5)=line(4:5);%px x
528                    Coord_file(i,1:3)=line(1:3);%phys x
529                end
530                eval(['x_' num2str(ifile+1) '=Coord_file(:,4:5)'';']);
531                eval(['x_' num2str(ifile+1) '(2,:)=ny-x_' num2str(ifile+1) '(2,:);' ]);
532                eval(['X_' num2str(ifile+1) '=Coord_file(:,1:3)'';']);
533                end
534            end
535        end
536    end
537end
538n_ima=numel(coord_files)+1;
539est_dist=[0;0;0;0;0];
540est_aspect_ratio=0;
541est_fc=[1;1];
542center_optim=0;
543run(fullfile(path_UVMAT,'toolbox_calib','go_calib_optim'));% apply fct 'toolbox_calib/go_calib_optim'
544GeometryCalib.CalibrationType='3D_linear';
545GeometryCalib.fx_fy=fc';
546GeometryCalib.Cx_Cy=cc';
547GeometryCalib.kc=kc(1);
548GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
549GeometryCalib.Tx_Ty_Tz=Tc_1';
550GeometryCalib.R=Rc_1;
551GeometryCalib.R(2,1:3)=-GeometryCalib.R(2,1:3);%inversion of the y image coordinate
552GeometryCalib.Tx_Ty_Tz(2)=-GeometryCalib.Tx_Ty_Tz(2);%inversion of the y image coordinate
553GeometryCalib.Cx_Cy(2)=ny-GeometryCalib.Cx_Cy(2);%inversion of the y image coordinate
554GeometryCalib.omc=(180/pi)*omc_1;%angles in degrees
555GeometryCalib.ErrorRMS=[];
556GeometryCalib.ErrorMax=[];
557
558%------------------------------------------------------------------------
559function GeometryCalib=calib_3D_quadr(Coord,handles)
560%------------------------------------------------------------------
561
562path_uvmat=which('uvmat');% check the path detected for source file uvmat
563path_UVMAT=fileparts(path_uvmat); %path to UVMAT
564huvmat=findobj(allchild(0),'Tag','uvmat');
565hhuvmat=guidata(huvmat);
566% check_cond=0;
567coord_files=get(handles.ListCoordFiles,'String');
568if ischar(coord_files)
569    coord_files={coord_files};
570end
571if isempty(coord_files{1}) || isequal(coord_files,{''})
572    coord_files={};
573end
574
575%retrieve the calibration points stored in the files listed in the popup list ListCoordFiles
576x_1=Coord(:,4:5)';%px coordinates of the ref points
577nx=str2num(get(hhuvmat.num_Npx,'String'));
578ny=str2num(get(hhuvmat.num_Npy,'String'));
579x_1(2,:)=ny-x_1(2,:);%reverse the y image coordinates
580X_1=Coord(:,1:3)';%phys coordinates of the ref points
581n_ima=numel(coord_files)+1;
582if ~isempty(coord_files)
583    msgbox_uvmat('CONFIRMATION',['The xy coordinates of the calibration points in ' num2str(n_ima) ' planes will be used'])
584    for ifile=1:numel(coord_files)
585    t=xmltree(coord_files{ifile});
586    s=convert(t);%convert to matlab structure
587        if isfield(s,'GeometryCalib')
588            if isfield(s.GeometryCalib,'SourceCalib')
589                if isfield(s.GeometryCalib.SourceCalib,'PointCoord')
590                PointCoord=s.GeometryCalib.SourceCalib.PointCoord;
591                Coord_file=zeros(length(PointCoord),5);%default
592                for i=1:length(PointCoord)
593                    line=str2num(PointCoord{i});
594                    Coord_file(i,4:5)=line(4:5);%px x
595                    Coord_file(i,1:3)=line(1:3);%phys x
596                end
597                eval(['x_' num2str(ifile+1) '=Coord_file(:,4:5)'';']);
598                eval(['x_' num2str(ifile+1) '(2,:)=ny-x_' num2str(ifile+1) '(2,:);' ]);
599                eval(['X_' num2str(ifile+1) '=Coord_file(:,1:3)'';']);
600                end
601            end
602        end
603    end
604end
605n_ima=numel(coord_files)+1;
606est_dist=[1;0;0;0;0];
607est_aspect_ratio=1;
608center_optim=0;
609run(fullfile(path_UVMAT,'toolbox_calib','go_calib_optim'));% apply fct 'toolbox_calib/go_calib_optim'
610
611GeometryCalib.CalibrationType='3D_quadr';
612GeometryCalib.fx_fy=fc';
613GeometryCalib.Cx_Cy=cc';
614GeometryCalib.kc=kc(1);
615GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
616GeometryCalib.Tx_Ty_Tz=Tc_1';
617if ~exist('Rc_1','var')
618    msgbox_uvmat('ERROR',['calibration function ' fullfile('toolbox_calib','go_calib_optim') ' did not converge: use multiple views or option 3D_extrinsic'])
619    return
620end
621GeometryCalib.R=Rc_1;
622GeometryCalib.R(2,1:3)=-GeometryCalib.R(2,1:3);%inversion of the y image coordinate
623GeometryCalib.Tx_Ty_Tz(2)=-GeometryCalib.Tx_Ty_Tz(2);%inversion of the y image coordinate
624GeometryCalib.Cx_Cy(2)=ny-GeometryCalib.Cx_Cy(2);%inversion of the y image coordinate
625GeometryCalib.omc=(180/pi)*omc_1;%angles in degrees
626GeometryCalib.ErrorRMS=[];
627GeometryCalib.ErrorMax=[];
628
629
630%------------------------------------------------------------------------
631function GeometryCalib=calib_3D_extrinsic(Coord,handles)
632%------------------------------------------------------------------
633path_uvmat=which('geometry_calib');% check the path detected for source file uvmat
634path_UVMAT=fileparts(path_uvmat); %path to UVMAT
635x_1=double(Coord(:,4:5)');%image coordiantes
636X_1=double(Coord(:,1:3)');% phys coordinates
637huvmat=findobj(allchild(0),'Tag','uvmat');
638hhuvmat=guidata(huvmat);
639ny=str2double(get(hhuvmat.num_Npy,'String'));
640x_1(2,:)=ny-x_1(2,:);%reverse the y image coordinates
641n_ima=1;
642GeometryCalib.CalibrationType='3D_extrinsic';
643GeometryCalib.fx_fy(1)=str2num(get(handles.fx,'String'));
644GeometryCalib.fx_fy(2)=str2num(get(handles.fy,'String'));
645GeometryCalib.Cx_Cy(1)=str2num(get(handles.Cx,'String'));
646GeometryCalib.Cx_Cy(2)=str2num(get(handles.Cy,'String'));
647GeometryCalib.kc=str2num(get(handles.kc,'String'));
648fct_path=fullfile(path_UVMAT,'toolbox_calib');
649addpath(fct_path)
650GeometryCalib.Cx_Cy(2)=ny-GeometryCalib.Cx_Cy(2);%reverse Cx_Cy(2) for calibration (inversion of px ordinate)
651[omc,Tc1,Rc1,H,x,ex,JJ] = compute_extrinsic(x_1,X_1,...
652   (GeometryCalib.fx_fy)',GeometryCalib.Cx_Cy',[GeometryCalib.kc 0 0 0 0]);
653rmpath(fct_path);
654GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
655GeometryCalib.Tx_Ty_Tz=Tc1';
656%inversion of z axis
657GeometryCalib.R=Rc1;
658GeometryCalib.R(2,1:3)=-GeometryCalib.R(2,1:3);%inversion of the y image coordinate
659GeometryCalib.Tx_Ty_Tz(2)=-GeometryCalib.Tx_Ty_Tz(2);%inversion of the y image coordinate
660GeometryCalib.Cx_Cy(2)=ny-GeometryCalib.Cx_Cy(2);%inversion of the y image coordinate
661GeometryCalib.omc=(180/pi)*omc';
662%GeometryCalib.R(3,1:3)=-GeometryCalib.R(3,1:3);%inversion for z upward
663
664
665
666%------------------------------------------------------------------------
667%function GeometryCalib=calib_tsai_heikkila(Coord)
668% TEST: NOT IMPLEMENTED
669%------------------------------------------------------------------
670% path_uvmat=which('uvmat');% check the path detected for source file uvmat
671% path_UVMAT=fileparts(path_uvmat); %path to UVMAT
672% path_calib=fullfile(path_UVMAT,'toolbox_calib_heikkila');
673% addpath(path_calib)
674% npoints=size(Coord,1);
675% Coord(:,1:3)=10*Coord(:,1:3);
676% Coord=[Coord zeros(npoints,2) -ones(npoints,1)];
677% [par,pos,iter,res,er,C]=cacal('dalsa',Coord);
678% GeometryCalib.CalibrationType='tsai';
679% GeometryCalib.focal=par(2);
680
681
682%------------------------------------------------------------------------
683% --- determine the rms of calibration error
684function ErrorRms=error_calib(calib_param,Calib,Coord)
685%calib_param: vector of free calibration parameters (to optimise)
686%Calib: structure of the given calibration parameters
687%Coord: list of phys coordinates (columns 1-3, and pixel coordinates (columns 4-5)
688Calib.f=25;
689Calib.dpx=0.012;
690Calib.dpy=0.012;
691Calib.sx=1;
692Calib.Cx=512;
693Calib.Cy=512;
694Calib.kappa1=calib_param(1);
695Calib.Tx=calib_param(2);
696Calib.Ty=calib_param(3);
697Calib.Tz=calib_param(4);
698alpha=calib_param(5);
699Calib.R=[cos(alpha) sin(alpha) 0;-sin(alpha) cos(alpha) 0;0 0 -1];
700
701X=Coord(:,1);
702Y=Coord(:,2);
703Z=Coord(:,3);
704x_ima=Coord(:,4);
705y_ima=Coord(:,5);
706[Xpoints,Ypoints]=px_XYZ(Calib,X,Y,Z);
707ErrorRms(1)=sqrt(mean((Xpoints-x_ima).*(Xpoints-x_ima)));
708ErrorRms(2)=sqrt(mean((Ypoints-y_ima).*(Ypoints-y_ima)));
709ErrorRms=mean(ErrorRms);
710
711% %------------------------------------------------------------------------
712% function XImage_Callback(hObject, eventdata, handles)
713% %------------------------------------------------------------------------
714% update_list(hObject, eventdata,handles)
715%
716% %------------------------------------------------------------------------
717% function YImage_Callback(hObject, eventdata, handles)
718% %------------------------------------------------------------------------
719% update_list(hObject, eventdata,handles)
720
721%------------------------------------------------------------------------
722% --- Executes on button press in STORE.
723function STORE_Callback(hObject, eventdata, handles)
724Coord=get(handles.ListCoord,'Data');
725%Object=read_geometry_calib(Coord_cell);
726unitlist=get(handles.CoordUnit,'String');
727unit=unitlist{get(handles.CoordUnit,'value')};
728GeometryCalib.CoordUnit=unit;
729GeometryCalib.SourceCalib.PointCoord=Coord(:,1:5);
730huvmat=findobj(allchild(0),'Name','uvmat');
731hhuvmat=guidata(huvmat);%handles of elements in the GUI uvmat
732% RootPath='';
733% RootFile='';
734if ~isempty(hhuvmat.RootPath)&& ~isempty(hhuvmat.RootFile)
735%     testhandle=1;
736    RootPath=get(hhuvmat.RootPath,'String');
737    RootFile=get(hhuvmat.RootFile,'String');
738    filebase=fullfile(RootPath,RootFile);
739    while exist([filebase '.xml'],'file')
740        filebase=[filebase '~'];
741    end
742    outputfile=[filebase '.xml'];
743    errormsg=update_imadoc(GeometryCalib,outputfile,'GeometryCalib');
744    if ~strcmp(errormsg,'')
745        msgbox_uvmat('ERROR',errormsg);
746    end
747    listfile=get(handles.ListCoordFiles,'string');
748    if isequal(listfile,{''})
749        listfile={outputfile};
750    else
751        listfile=[listfile;{outputfile}];%update the list of coord files
752    end
753    set(handles.ListCoordFiles,'string',listfile);
754end
755set(handles.ListCoord,'Data',[])
756
757% --------------------------------------------------------------------
758% --- Executes on button press in CLEAR_PTS: clear the list of calibration points
759function CLEAR_PTS_Callback(hObject, eventdata, handles)
760% --------------------------------------------------------------------
761set(handles.ListCoord,'Data',[])
762PLOT_Callback(hObject, eventdata, handles)
763
764%------------------------------------------------------------------------
765% --- Executes on button press in CLEAR.
766function CLEAR_Callback(hObject, eventdata, handles)
767%------------------------------------------------------------------------
768set(handles.ListCoordFiles,'Value',1)
769set(handles.ListCoordFiles,'String',{''})
770
771%------------------------------------------------------------------------
772% --- Executes on selection change in CheckEnableMouse.
773function CheckEnableMouse_Callback(hObject, eventdata, handles)
774%------------------------------------------------------------------------
775choice=get(handles.CheckEnableMouse,'Value');
776if choice
777    set(handles.CheckEnableMouse,'BackgroundColor',[1 1 0])
778    huvmat=findobj(allchild(0),'tag','uvmat');
779    if ishandle(huvmat)
780        hhuvmat=guidata(huvmat);
781        if get(hhuvmat.CheckEditObject,'Value')
782        set(hhuvmat.CheckEditObject,'Value',0)
783        uvmat('CheckEditObject_Callback',hhuvmat.CheckEditObject,[],hhuvmat)
784        end
785    end
786else
787    set(handles.CheckEnableMouse,'BackgroundColor',[0.7 0.7 0.7])
788end
789
790
791% --------------------------------------------------------------------
792function MenuHelp_Callback(hObject, eventdata, handles)
793path_to_uvmat=which('uvmat');% check the path of uvmat
794pathelp=fileparts(path_to_uvmat);
795helpfile=fullfile(pathelp,'uvmat_doc','uvmat_doc.html');
796if isempty(dir(helpfile)), msgbox_uvmat('ERROR','Please put the help file uvmat_doc.html in the sub-directory /uvmat_doc of the UVMAT package')
797else
798   addpath (fullfile(pathelp,'uvmat_doc'))
799   web([helpfile '#geometry_calib'])
800end
801
802% --------------------------------------------------------------------
803function MenuSetScale_Callback(hObject, eventdata, handles)
804
805 answer=msgbox_uvmat('INPUT_TXT','scale pixel/cm?','');
806 %create test points
807 huvmat=findobj(allchild(0),'Name','uvmat');%find the current uvmat interface handle
808UvData=get(huvmat,'UserData');%Data associated to the current uvmat interface
809npy=size(UvData.Field.A,1);
810npx=size(UvData.Field.A,2);
811Xima=[0.25*npx 0.75*npx 0.75*npx 0.25*npx]';
812Yima=[0.25*npy 0.25*npy 0.75*npy 0.75*npy]';
813x=Xima/str2num(answer);
814y=Yima/str2num(answer);
815Coord=[x y zeros(4,1) Xima Yima zeros(4,1)];
816set(handles.ListCoord,'Data',Coord)
817set(handles.APPLY,'BackgroundColor',[1 0 1])
818
819%------------------------------------------------------------------------
820function MenuCreateGrid_Callback(hObject, eventdata, handles)
821%------------------------------------------------------------------------
822CalibData=get(handles.geometry_calib,'UserData');
823Tinput=[];%default
824if isfield(CalibData,'grid')
825    Tinput=CalibData.grid;
826end
827[T,CalibData.grid]=create_grid(Tinput);%display the GUI create_grid
828set(handles.geometry_calib,'UserData',CalibData)
829
830%grid in phys space
831Coord=get(handles.ListCoord,'Data');
832Coord(1:size(T,1),1:3)=T;%update the existing list of phys coordinates from the GUI create_grid
833set(handles.ListCoord,'Data',Coord)
834set(handles.APPLY,'BackgroundColor',[1 0 1])
835
836% -----------------------------------------------------------------------
837% --- automatic grid dectection from local maxima of the images
838function MenuDetectGrid_Callback(hObject, eventdata, handles)
839%------------------------------------------------------------------------
840%% read the four last point coordinates in pixels
841Coord=get(handles.ListCoord,'Data');%read list of coordinates on geometry_calib
842nbpoints=size(Coord,1); %nbre of calibration points
843if nbpoints~=4
844    msgbox_uvmat('ERROR','four points must have be selected by the mouse to delimitate the phys grid area; the Ox axis will be defined by the two first points')
845    return
846end
847corners_X=(Coord(end:-1:end-3,4)); %pixel absissa of the four corners
848corners_Y=(Coord(end:-1:end-3,5));
849
850%reorder the last two points (the two first in the list) if needed
851angles=angle((corners_X-corners_X(1))+1i*(corners_Y-corners_Y(1)));
852if abs(angles(4)-angles(2))>abs(angles(3)-angles(2))
853      X_end=corners_X(4);
854      Y_end=corners_Y(4);
855      corners_X(4)=corners_X(3);
856      corners_Y(4)=corners_Y(3);
857      corners_X(3)=X_end;
858      corners_Y(3)=Y_end;
859end
860
861%% initiate the grid
862CalibData=get(handles.geometry_calib,'UserData');%get information stored on the GUI geometry_calib
863grid_input=[];%default
864if isfield(CalibData,'grid')
865    grid_input=CalibData.grid;%retrieve the previously used grid
866end
867[T,CalibData.grid,CalibData.grid.CheckWhite]=create_grid(grid_input,'detect_grid');%display the GUI create_grid, read the set of phys coordinates T
868set(handles.geometry_calib,'UserData',CalibData)%store the phys grid parameters for later use
869
870%% read the current image, displayed in the GUI uvmat
871huvmat=findobj(allchild(0),'Name','uvmat');
872UvData=get(huvmat,'UserData');
873A=UvData.Field.A;%currently displayed image
874npxy=size(A);
875X=[CalibData.grid.x_0 CalibData.grid.x_1 CalibData.grid.x_0 CalibData.grid.x_1]';%corner absissa in the phys coordinates (cm)
876Y=[CalibData.grid.y_0 CalibData.grid.y_0 CalibData.grid.y_1 CalibData.grid.y_1]';%corner ordinates in the phys coordinates (cm)
877
878%calculate transform matrices for plane projection: rectangle assumed to be viewed in perspective
879% reference: http://alumni.media.mit.edu/~cwren/interpolator/ by Christopher R. Wren
880B = [ X Y ones(size(X)) zeros(4,3)        -X.*corners_X -Y.*corners_X ...
881      zeros(4,3)        X Y ones(size(X)) -X.*corners_Y -Y.*corners_Y ];
882B = reshape (B', 8 , 8 )';
883D = [ corners_X , corners_Y ];
884D = reshape (D', 8 , 1 );
885l = (B' * B)\B' * D;
886Amat = reshape([l(1:6)' 0 0 1 ],3,3)';
887C = [l(7:8)' 1];
888
889% transform grid image into 'phys' coordinates
890GeometryCalib.CalibrationType='3D_linear';
891GeometryCalib.fx_fy=[1 1];
892GeometryCalib.Tx_Ty_Tz=[Amat(1,3) Amat(2,3) 1];
893GeometryCalib.R=[Amat(1,1),Amat(1,2),0;Amat(2,1),Amat(2,2),0;C(1),C(2),0];
894GeometryCalib.CoordUnit='cm';
895path_uvmat=which('uvmat');% check the path detected for source file uvmat
896path_UVMAT=fileparts(path_uvmat); %path to UVMAT
897addpath(fullfile(path_UVMAT,'transform_field'))
898Data.ListVarName={'AY','AX','A'};
899Data.VarDimName={'AY','AX',{'AY','AX'}};
900if ndims(A)==3
901    A=mean(A,3);
902end
903Data.A=A-min(min(A));
904Data.AY=[npxy(1)-0.5 0.5];
905Data.AX=[0.5 npxy(2)];
906Data.CoordUnit='pixel';
907Calib.GeometryCalib=GeometryCalib;
908DataOut=phys(Data,Calib);
909rmpath(fullfile(path_UVMAT,'transform_field'))
910Amod=DataOut.A;% current imgage expressed in 'phys' coord
911Rangx=DataOut.AX;
912Rangy=DataOut.AY;
913if CalibData.grid.CheckWhite
914    Amod=double(Amod);%case of white grid markers: will look for image maxima
915else
916    Amod=-double(Amod);%case of black grid markers: will look for image minima
917end
918
919%% detection of local image extrema in each direction
920Dx=(Rangx(2)-Rangx(1))/(npxy(2)-1); %x mesh in real space
921Dy=(Rangy(2)-Rangy(1))/(npxy(1)-1); %y mesh in real space
922ind_range_x=ceil(abs(GeometryCalib.R(1,1)*CalibData.grid.Dx/3));% range of search of image ma around each point obtained by linear interpolation from the marked points
923ind_range_y=ceil(abs(GeometryCalib.R(2,2)*CalibData.grid.Dy/3));% range of search of image ma around each point obtained by linear interpolation from the marked points
924nbpoints=size(T,1);
925%lokk for image maxima around each expected pgrid point
926for ipoint=1:nbpoints
927    i0=1+round((T(ipoint,1)-Rangx(1))/Dx);%round(Xpx(ipoint));
928    j0=1+round((T(ipoint,2)-Rangy(1))/Dy);%round(Xpx(ipoint));
929    j0min=max(j0-ind_range_y,1);
930    j0max=min(j0+ind_range_y,size(Amod,1));
931    i0min=max(i0-ind_range_x,1);
932    i0max=min(i0+ind_range_x,size(Amod,2));
933    Asub=Amod(j0min:j0max,i0min:i0max);
934 
935
936   
937    x_profile=sum(Asub,1);%profile of subimage summed over y
938    y_profile=sum(Asub,2);%profile of subimage summed over x
939    %%%%
940%     if ipoint==5
941%                 figure(10)
942%   imagesc(Asub)
943%     figure(11)
944%     plot(x_profile,'r')
945%     hold on
946%     plot(y_profile,'b')
947%     end
948    %%%%
949    [tild,ind_x_max]=max(x_profile);
950    [tild,ind_y_max]=max(y_profile);
951    %sub-pixel improvement using moments
952    x_shift=0;
953    y_shift=0;
954    if ind_x_max+2<=numel(x_profile) && ind_x_max-2>=1
955        Atop=x_profile(ind_x_max-2:ind_x_max+2);% extract x profile around the max
956        x_shift=sum(Atop.*[-2 -1 0 1 2])/sum(Atop);
957    end
958    if ind_y_max+2<=numel(y_profile) && ind_y_max-2>=1
959        Atop=y_profile(ind_y_max-2:ind_y_max+2);% extract y profile around the max
960        y_shift=sum(Atop.*[-2 -1 0 1 2]')/sum(Atop);
961    end
962    Delta(ipoint,1)=(i0min+ind_x_max-1+x_shift-i0)*Dx;%shift from the initial guess
963    Delta(ipoint,2)=(j0min+ind_y_max-1+y_shift-j0)*Dy;
964end
965Tmod=T(:,(1:2))+Delta;% 'phys' coordinates of the detected points
966Tmod(:,2)=flipdim(Tmod(:,2),1);% inverse the order of y coordinates
967[Xpx,Ypx]=px_XYZ(GeometryCalib,Tmod(:,1),Tmod(:,2));% image coordinates of the detected points
968Coord=[T Xpx Ypx zeros(size(T,1),1)];
969set(handles.ListCoord,'Data',Coord)
970PLOT_Callback(hObject, eventdata, handles)
971set(handles.APPLY,'BackgroundColor',[1 0 1])
972
973%-----------------------------------------------------------------------
974function MenuTranslatePoints_Callback(hObject, eventdata, handles)
975%-----------------------------------------------------------------------
976%hcalib=get(handles.calib_type,'parent');%handles of the GUI geometry_calib
977CalibData=get(handles.geometry_calib,'UserData');
978Tinput=[];%default
979if isfield(CalibData,'translate')
980    Tinput=CalibData.translate;
981end
982T=translate_points(Tinput);%display translate_points GUI and get shift parameters
983CalibData.translate=T;
984set(handles.geometry_calib,'UserData',CalibData)
985%translation
986Coord=get(handles.ListCoord,'Data');
987Coord(:,1)=T(1)+Coord(:,1);
988Coord(:,2)=T(2)+Coord(:,2);
989Coord(:,3)=T(3)+Coord(:,3);
990set(handles.ListCoord,'Data',Coord);
991set(handles.APPLY,'BackgroundColor',[1 0 1])
992
993% --------------------------------------------------------------------
994function MenuRotatePoints_Callback(hObject, eventdata, handles)
995%hcalib=get(handles.calib_type,'parent');%handles of the GUI geometry_calib
996CalibData=get(handles.geometry_calib,'UserData');
997Tinput=[];%default
998if isfield(CalibData,'rotate')
999    Tinput=CalibData.rotate;
1000end
1001T=rotate_points(Tinput);%display rotate_points GUI to introduce rotation parameters
1002CalibData.rotate=T;
1003set(handles.geometry_calib,'UserData',CalibData)
1004%-----------------------------------------------------
1005%rotation
1006Phi=T(1);
1007O_x=0;%default
1008O_y=0;%default
1009if numel(T)>=2
1010    O_x=T(2);%default
1011end
1012if numel(T)>=3
1013    O_y=T(3);%default
1014end
1015Coord=get(handles.ListCoord,'Data');
1016r1=cos(pi*Phi/180);
1017r2=-sin(pi*Phi/180);
1018r3=sin(pi*Phi/180);
1019r4=cos(pi*Phi/180);
1020x=Coord(:,1)-O_x;
1021y=Coord(:,2)-O_y;
1022Coord(:,1)=r1*x+r2*y;
1023Coord(:,2)=r3*x+r4*y;
1024set(handles.ListCoord,'Data',Coord)
1025set(handles.APPLY,'BackgroundColor',[1 0 1])
1026
1027% --------------------------------------------------------------------
1028function MenuImportPoints_Callback(hObject, eventdata, handles)
1029fileinput=browse_xml(hObject, eventdata, handles);
1030if isempty(fileinput)
1031    return
1032end
1033[s,errormsg]=imadoc2struct(fileinput,'GeometryCalib');
1034if ~isfield(s,'GeometryCalib')
1035    msgbox_uvmat('ERROR','invalid input file: no geometry_calib data')
1036    return
1037end
1038GeometryCalib=s.GeometryCalib;
1039if ~(isfield(GeometryCalib,'SourceCalib')&&isfield(GeometryCalib.SourceCalib,'PointCoord'))
1040        msgbox_uvmat('ERROR','invalid input file: no calibration points')
1041    return
1042end
1043Coord=GeometryCalib.SourceCalib.PointCoord;
1044Coord=[Coord zeros(size(Coord,1),1)];
1045set(handles.ListCoord,'Data',Coord)
1046PLOT_Callback(handles.geometry_calib, [], handles)
1047set(handles.APPLY,'BackgroundColor',[1 0 1])
1048
1049% -----------------------------------------------------------------------
1050function MenuImportIntrinsic_Callback(hObject, eventdata, handles)
1051%------------------------------------------------------------------------
1052fileinput=browse_xml(hObject, eventdata, handles);
1053if isempty(fileinput)
1054    return
1055end
1056[s,errormsg]=imadoc2struct(fileinput,'GeometryCalib');
1057GeometryCalib=s.GeometryCalib;
1058display_intrinsic(GeometryCalib,handles)
1059
1060% -----------------------------------------------------------------------
1061function MenuImportAll_Callback(hObject, eventdata, handles)
1062%------------------------------------------------------------------------
1063fileinput=browse_xml(hObject, eventdata, handles);
1064if ~isempty(fileinput)
1065    loadfile(handles,fileinput)
1066end
1067
1068% -----------------------------------------------------------------------
1069% --- Executes on menubar option Import/Grid file: introduce previous grid files
1070function MenuGridFile_Callback(hObject, eventdata, handles)
1071% -----------------------------------------------------------------------
1072inputfile=browse_xml(hObject, eventdata, handles);
1073listfile=get(handles.ListCoordFiles,'String');
1074if isequal(listfile,{''})
1075    listfile={inputfile};
1076else
1077    listfile=[listfile;{inputfile}];%update the list of coord files
1078end
1079set(handles.ListCoordFiles,'string',listfile);
1080
1081
1082%------------------------------------------------------------------------
1083function fileinput=browse_xml(hObject, eventdata, handles)
1084%------------------------------------------------------------------------
1085fileinput=[];%default
1086oldfile=''; %default
1087UserData=get(handles.geometry_calib,'UserData');
1088if isfield(UserData,'XmlInputFile')
1089    oldfile=UserData.XmlInputFile;
1090end
1091[FileName, PathName, filterindex] = uigetfile( ...
1092       {'*.xml;*.mat', ' (*.xml,*.mat)';
1093       '*.xml',  '.xml files '; ...
1094        '*.mat',  '.mat matlab files '}, ...
1095        'Pick a file',oldfile);
1096fileinput=[PathName FileName];%complete file name
1097testblank=findstr(fileinput,' ');%look for blanks
1098if ~isempty(testblank)
1099    msgbox_uvmat('ERROR','forbidden input file name or path: no blank character allowed')
1100    return
1101end
1102sizf=size(fileinput);
1103if (~ischar(fileinput)||~isequal(sizf(1),1)),return;end
1104UserData.XmlInputFile=fileinput;
1105set(handles.geometry_calib,'UserData',UserData)%record current file foer further use of browser
1106
1107% -----------------------------------------------------------------------
1108function Heading=loadfile(handles,fileinput)
1109%------------------------------------------------------------------------
1110Heading=[];%default
1111[s,errormsg]=imadoc2struct(fileinput,'Heading','GeometryCalib');
1112if ~isempty(errormsg)
1113    msgbox_uvmat('ERROR',errormsg)
1114    return
1115end
1116if ~isempty(s.Heading)
1117    Heading=s.Heading;
1118end
1119
1120GeometryCalib=s.GeometryCalib;
1121fx=1;fy=1;Cx=0;Cy=0;kc=0; %default
1122CoordCell={};
1123Tabchar={};%default
1124val_cal=1;%default
1125if ~isempty(GeometryCalib)
1126    % choose the calibration option
1127    if isfield(GeometryCalib,'CalibrationType')
1128        calib_list=get(handles.calib_type,'String');
1129        for ilist=1:numel(calib_list)
1130            if strcmp(calib_list{ilist},GeometryCalib.CalibrationType)
1131                val_cal=ilist;
1132                break
1133            end
1134        end
1135    end
1136    display_intrinsic(GeometryCalib,handles)%intrinsic param
1137    %extrinsic param
1138    if isfield(GeometryCalib,'Tx_Ty_Tz')
1139        Tx_Ty_Tz=GeometryCalib.Tx_Ty_Tz;
1140        set(handles.Tx,'String',num2str(GeometryCalib.Tx_Ty_Tz(1),4))
1141        set(handles.Ty,'String',num2str(GeometryCalib.Tx_Ty_Tz(2),4))
1142        set(handles.Tz,'String',num2str(GeometryCalib.Tx_Ty_Tz(3),4))
1143    end
1144    if isfield(GeometryCalib,'omc')
1145        set(handles.Phi,'String',num2str(GeometryCalib.omc(1),4))
1146        set(handles.Theta,'String',num2str(GeometryCalib.omc(2),4))
1147        set(handles.Psi,'String',num2str(GeometryCalib.omc(3),4))
1148    end
1149    if isfield(GeometryCalib,'SourceCalib')
1150        calib=GeometryCalib.SourceCalib.PointCoord;
1151        Coord=[calib zeros(size(calib,1),1)];
1152        set(handles.ListCoord,'Data',Coord)
1153    end
1154    PLOT_Callback(handles.geometry_calib, [], handles)
1155    set(handles.APPLY,'BackgroundColor',[1 0 1])
1156end
1157set(handles.calib_type,'Value',val_cal)
1158
1159if isempty(CoordCell)% allow mouse action by default in the absence of input points
1160    set(handles.CheckEnableMouse,'Value',1)
1161    set(handles.CheckEnableMouse,'BackgroundColor',[1 1 0])
1162else % does not allow mouse action by default in the presence of input points
1163    set(handles.CheckEnableMouse,'Value',0)
1164    set(handles.CheckEnableMouse,'BackgroundColor',[0.7 0.7 0.7])
1165end
1166
1167%------------------------------------------------------------------------
1168%---display calibration intrinsic parameters
1169function display_intrinsic(GeometryCalib,handles)
1170%------------------------------------------------------------------------
1171fx=[];
1172fy=[];
1173if isfield(GeometryCalib,'fx_fy')
1174    fx=GeometryCalib.fx_fy(1);
1175    fy=GeometryCalib.fx_fy(2);
1176end
1177Cx_Cy=[0 0];%default
1178if isfield(GeometryCalib,'Cx_Cy')
1179    Cx_Cy=GeometryCalib.Cx_Cy;
1180end
1181kc=0;
1182if isfield(GeometryCalib,'kc')
1183    kc=GeometryCalib.kc; %* GeometryCalib.focal*GeometryCalib.focal;
1184end
1185set(handles.fx,'String',num2str(fx,5))
1186set(handles.fy,'String',num2str(fy,5))
1187set(handles.Cx,'String',num2str(Cx_Cy(1),'%1.1f'))
1188set(handles.Cy,'String',num2str(Cx_Cy(2),'%1.1f'))
1189set(handles.kc,'String',num2str(kc,'%1.4f'))
1190
1191
1192% --- Executes when user attempts to close geometry_calib.
1193function geometry_calib_CloseRequestFcn(hObject, eventdata, handles)
1194
1195delete(hObject); % closes the figure
1196
1197%------------------------------------------------------------------------
1198% --- Executes on button press in PLOT.
1199%------------------------------------------------------------------------
1200function PLOT_Callback(hObject, eventdata, handles)
1201huvmat=findobj(allchild(0),'Name','uvmat');%find the current uvmat interface handle
1202hhuvmat=guidata(huvmat); %handles of GUI elements in uvmat
1203h_menu_coord=findobj(huvmat,'Tag','TransformName');
1204menu=get(h_menu_coord,'String');
1205choice=get(h_menu_coord,'Value');
1206if iscell(menu)
1207    option=menu{choice};
1208else
1209    option='px'; %default
1210end
1211Coord=get(handles.ListCoord,'Data');
1212if ~isempty(Coord)
1213    if isequal(option,'phys')
1214        Coord_plot=Coord(:,1:3);
1215    elseif isequal(option,'px')||isequal(option,'')
1216        Coord_plot=Coord(:,4:5);
1217    else
1218        msgbox_uvmat('ERROR','the choice in menu_coord of uvmat must be blank, px or phys ')
1219    end
1220end
1221
1222set(0,'CurrentFigure',huvmat)
1223set(huvmat,'CurrentAxes',hhuvmat.PlotAxes)
1224hh=findobj('Tag','calib_points');
1225if  ~isempty(Coord) && isempty(hh)
1226    hh=line(Coord_plot(:,1),Coord_plot(:,2),'Color','m','Tag','calib_points','LineStyle','.','Marker','+');
1227elseif isempty(Coord)%empty list of points, suppress the plot
1228    delete(hh)
1229else
1230    set(hh,'XData',Coord_plot(:,1))
1231    set(hh,'YData',Coord_plot(:,2))
1232end
1233pause(.1)
1234figure(handles.geometry_calib)
1235
1236%------------------------------------------------------------------------
1237% --- Executes on button press in Copy: display Coord on the Matlab work space
1238%------------------------------------------------------------------------
1239function Copy_Callback(hObject, eventdata, handles)
1240global Coord
1241evalin('base','global Coord')%make CurData global in the workspace
1242Coord=get(handles.ListCoord,'Data');
1243display('coordinates of calibration points (phys,px,marker) :')
1244evalin('base','Coord') %display CurData in the workspace
1245commandwindow; %brings the Matlab command window to the front
1246
1247%------------------------------------------------------------------------
1248% --- Executes when selected cell(s) is changed in ListCoord.
1249%------------------------------------------------------------------------
1250function ListCoord_CellSelectionCallback(hObject, eventdata, handles)
1251if ~isempty(eventdata.Indices)
1252    iline=eventdata.Indices(1);% selected line number
1253    Data=get(handles.ListCoord,'Data');
1254    Data(:,6)=zeros(size(Data,1),1);
1255    Data(iline,6)=-1;% mark the selected line
1256    set(handles.ListCoord,'Data',Data)
1257    update_calib_marker(Data(iline,:))
1258end
1259
1260%------------------------------------------------------------------------
1261% --- Executes when entered data in editable cell(s) in ListCoord.
1262%------------------------------------------------------------------------
1263function ListCoord_CellEditCallback(hObject, eventdata, handles)
1264
1265Input=str2num(eventdata.EditData);%pasted input
1266Coord=get(handles.ListCoord,'Data');
1267iline=eventdata.Indices(1);% selected line number
1268if size(Coord,1)<iline+numel(Input)
1269    Coord=[Coord ; zeros(iline+numel(Input)-size(Coord,1),6)];% append zeros to fit the new column
1270end
1271Coord(iline:iline+numel(Input)-1,eventdata.Indices(2))=Input';
1272set(handles.ListCoord,'Data',Coord)
1273PLOT_Callback(hObject, eventdata, handles)
1274
1275%------------------------------------------------------------------------
1276% --- 'key_press_fcn:' function activated when a key is pressed on the keyboard
1277%------------------------------------------------------------------------
1278function ListCoord_KeyPressFcn(hObject, eventdata, handles)
1279xx=double(get(handles.geometry_calib,'CurrentCharacter'));%get the keyboard character
1280if ismember(xx,[30 31 127])% arrow upward, downward, or delete
1281    Coord=get(handles.ListCoord,'Data');
1282    ind=find(Coord(:,6));%find the marker '+' for line selection
1283    Coord(:,6)=zeros(size(Coord,1),1);% desactivate the current line mark
1284    switch xx
1285        case 30 % arrow upward
1286            Coord(ind-1,6)=1;
1287        case 31% arrow downward
1288            Coord(ind+1,6)=1;
1289        case 127% remove line
1290            Coord(ind,:)=[];
1291            PLOT_Callback(hObject,eventdata,handles)
1292            set(handles.APPLY,'BackgroundColor',[1 0 1])
1293        otherwise
1294    end
1295    set(handles.ListCoord,'Data',Coord);
1296else
1297    set(handles.APPLY,'BackgroundColor',[1 0 1])
1298end
1299
1300
1301%------------------------------------------------------------------------
1302% --- update the plot of calibration points
1303%------------------------------------------------------------------------
1304function update_calib_marker(Coord)
1305%% update the plot on uvmat
1306huvmat=findobj(allchild(0),'Name','uvmat');%find the current uvmat interface handle
1307hplot=findobj(huvmat,'Tag','PlotAxes');%main plotting axis of uvmat
1308hhh=findobj(hplot,'Tag','calib_marker');
1309
1310h_menu_coord=findobj(huvmat,'Tag','TransformName');
1311menu=get(h_menu_coord,'String');
1312choice=get(h_menu_coord,'Value');
1313if iscell(menu)
1314    option=menu{choice};
1315else
1316    option='px'; %default
1317end
1318if isequal(option,'phys')
1319    XCoord=Coord(1);
1320    YCoord=Coord(2);
1321elseif isequal(option,'px')|| isequal(option,'')
1322    XCoord=Coord(4);
1323    YCoord=Coord(5);
1324else
1325    msgbox_uvmat('ERROR','the choice in menu_coord of uvmat must be blank, px or phys ')
1326end
1327if isempty(XCoord)||isempty(YCoord)
1328     if ~isempty(hhh)
1329        delete(hhh)%delete the circle marker
1330    end
1331    return
1332end
1333xlim=get(hplot,'XLim');
1334ylim=get(hplot,'YLim');
1335ind_range=max(abs(xlim(2)-xlim(1)),abs(ylim(end)-ylim(1)))/20;%defines the size of the circle marker
1336if isempty(hhh)
1337    set(0,'CurrentFig',huvmat)
1338    set(huvmat,'CurrentAxes',hplot)
1339    rectangle('Curvature',[1 1],...
1340              'Position',[XCoord-ind_range/2 YCoord-ind_range/2 ind_range ind_range],'EdgeColor','m',...
1341              'LineStyle','-','Tag','calib_marker');
1342else
1343    set(hhh,'Position',[XCoord-ind_range/2 YCoord-ind_range/2 ind_range ind_range])
1344end
1345
1346%------------------------------------------------------------------------
1347% --- Executes on button press in ClearLine: remove the selected line in the table Coord
1348%------------------------------------------------------------------------
1349function ClearLine_Callback(hObject, eventdata, handles)
1350
1351Coord=get(handles.ListCoord,'Data');
1352ind=find(Coord(:,6));%find the marker '-' for line selection
1353if isempty(ind)
1354    msgbox_uvmat('WARNING','no line suppressed, select a line in the table')
1355else
1356    answer=msgbox_uvmat('INPUT_Y-N',['suppress line ' num2str(ind) '?']);
1357    if isequal(answer,'Yes')
1358Coord(:,6)=zeros(size(Coord,1),1);% desactivate the current line mark
1359Coord(ind,:)=[];
1360PLOT_Callback(hObject,eventdata,handles)
1361set(handles.APPLY,'BackgroundColor',[1 0 1])
1362set(handles.ListCoord,'Data',Coord);
1363    end
1364end
Note: See TracBrowser for help on using the repository browser.