source: trunk/src/geometry_calib.m @ 128

Last change on this file since 128 was 128, checked in by sommeria, 13 years ago

series: give writting access to the group for all subdirectories produced
uvmat.fig: change of vect and scalar frames (to be consistent with view_field)
uvmat: various cleaning
plot_field: various cleaning to improve axes definition and avoid blinking
geometry_calib: improved dispay of point coordiantes, improved link with dataview for REPLICATE.
struct2nc: repair bug , file was not closed.
cell2tab: cleaning
dataview: improve the browser
civ: solve pb of image naming

File size: 62.2 KB
RevLine 
[121]1%%
[2]2%'geometry_calib': performs geometric calibration from a set of reference points
3%
4% function varargout = geometry_calib(varargin)
5%
[109]6%A%AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
[2]7%  Copyright Joel Sommeria, 2008, LEGI / CNRS-UJF-INPG, sommeria@coriolis-legi.org.
8%AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
9%     This file is part of the toolbox UVMAT.
10%
11%     UVMAT is free software; you can redistribute it and/or modify
12%     it under the terms of the GNU General Public License as published by
13%     the Free Software Foundation; either version 2 of the License, or
14%     (at your option) any later version.
15%
16%     UVMAT is distributed in the hope that it will be useful,
17%     but WITHOUT ANY WARRANTY; without even the implied warranty of
18%     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
19%     GNU General Public License (file UVMAT/COPYING.txt) for more details.
20%AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
21
22function varargout = geometry_calib(varargin)
23% GEOMETRY_CALIB M-file for geometry_calib.fig
24%      GEOMETRY_CALIB, by itself, creates a MenuCoord GEOMETRY_CALIB or raises the existing
25%      singleton*.
26%
27%      H = GEOMETRY_CALIB returns the handle to a MenuCoord GEOMETRY_CALIB or the handle to
28%      the existing singleton*.
29%
30%      GEOMETRY_CALIB('CALLBACK',hObject,eventData,handles,...) calls the local
31%      function named CALLBACK in GEOMETRY_CALIB.M with the given input arguments.
32%
33%      GEOMETRY_CALIB('Property','Value',...) creates a MenuCoord GEOMETRY_CALIB or raises the
34%      existing singleton*.  Starting from the left, property value pairs are
35%      applied to the GUI before geometry_calib_OpeningFunction gets called.  An
36%      unrecognized property name or invalid value makes property application
37%      stop.  All inputs are passed to geometry_calib_OpeningFcn via varargin.
38%
39%      *See GUI Options on GUIDE's Tools menu.  Choose "GUI allows only one
40%      instance to run (singleton)".
41%
42% See also: GUIDE, GUIDATA, GUIHANDLES
43
44% Edit the above text to modify the response to help geometry_calib
45
[114]46% Last Modified by GUIDE v2.5 05-Oct-2010 13:47:00
[2]47
48% Begin initialization code - DO NOT edit
49gui_Singleton = 1;
50gui_State = struct('gui_Name',       mfilename, ...
51                   'gui_Singleton',  gui_Singleton, ...
52                   'gui_OpeningFcn', @geometry_calib_OpeningFcn, ...
53                   'gui_OutputFcn',  @geometry_calib_OutputFcn, ...
54                   'gui_LayoutFcn',  [] , ...
55                   'gui_Callback',   []);
[128]56if nargin && ischar(varargin{1}) && ~isempty(regexp(varargin{1},'_Callback','once'))
[2]57    gui_State.gui_Callback = str2func(varargin{1});
58end
59
60if nargout
61    [varargout{1:nargout}] = gui_mainfcn(gui_State, varargin{:});
62else
63    gui_mainfcn(gui_State, varargin{:});
64end
65% End initialization code - DO NOT edit
66
67
68% --- Executes just before geometry_calib is made visible.
69%INPUT:
70%handles: handles of the geometry_calib interface elements
71% PlotHandles: set of handles of the elements contolling the plotting
72% parameters on the uvmat interface (obtained by 'get_plot_handle.m')
[60]73%------------------------------------------------------------------------
[116]74function geometry_calib_OpeningFcn(hObject, eventdata, handles,inputfile,pos)
[60]75%------------------------------------------------------------------------
[2]76% Choose default command line output for geometry_calib
77handles.output = hObject;
78
79% Update handles structure
80guidata(hObject, handles);
[71]81set(hObject,'DeleteFcn',{@closefcn})%
82
[2]83%set the position of the interface
[116]84if exist('pos','var')&& length(pos)>2
[2]85    pos_gui=get(hObject,'Position');
86    pos_gui(1)=pos(1);
87    pos_gui(2)=pos(2);
88    set(hObject,'Position',pos_gui);
89end
[109]90
91%set menu of calibration options
92%set(handles.calib_type,'String',{'rescale';'linear';'perspective';'normal';'tsai';'bouguet';'extrinsic'})
[121]93set(handles.calib_type,'String',{'rescale';'linear';'3D_linear';'3D_quadr';'3D_extrinsic'})
[2]94inputxml='';
[116]95if exist('inputfile','var')&& ~isempty(inputfile)
[114]96    struct.XmlInputFile=inputfile;
[128]97    [Pathsub,RootFile,field_count,str2,str_a,str_b,ext]=name2display(inputfile);
[116]98    if ~strcmp(ext,'.xml')
[121]99        inputfile=[fullfile(Pathsub,RootFile) '.xml'];%xml file corresponding to the input file
[116]100    end
[2]101end
[71]102set(handles.ListCoord,'String',{'......'})
[116]103if exist(inputfile,'file')
[128]104    Heading=loadfile(handles,inputfile);% load the point coordiantes existing in the xml file
[2]105end
[128]106if isfield(Heading,'Campaign')&& ischar(Heading.Campaign)
107    struct.Campaign=Heading.Campaign;
108end
109set(hObject,'UserData',struct)
[2]110set(handles.ListCoord,'KeyPressFcn',{@key_press_fcn,handles})%set keyboard action function
111
[71]112
[17]113%------------------------------------------------------------------------
[2]114% --- Outputs from this function are returned to the command line.
115function varargout = geometry_calib_OutputFcn(hObject, eventdata, handles)
[17]116%------------------------------------------------------------------------
[2]117% Get default command line output from handles structure
118varargout{1} = handles.output;
119varargout{2}=handles;
[67]120%
[60]121%------------------------------------------------------------------------
[2]122% executed when closing: set the parent interface button to value 0
[67]123function closefcn(gcbo,eventdata)
[60]124%------------------------------------------------------------------------
[2]125huvmat=findobj(allchild(0),'Name','uvmat');
[67]126if ~isempty(huvmat)
127    handles=guidata(huvmat);
[71]128    set(handles.MenuMask,'enable','on')
129    set(handles.MenuGrid,'enable','on')
[67]130    set(handles.MenuObject,'enable','on')
131    set(handles.MenuEdit,'enable','on')
132    set(handles.edit,'enable','on')
133    hobject=findobj(handles.axes3,'tag','calib_points');
134    if ~isempty(hobject)
135        delete(hobject)
136    end
137    hobject=findobj(handles.axes3,'tag','calib_marker');
138    if ~isempty(hobject)
139        delete(hobject)
140    end   
[2]141end
142
[60]143%------------------------------------------------------------------------
[2]144% --- Executes on button press in calibrate_lin.
145function APPLY_Callback(hObject, eventdata, handles)
[60]146%------------------------------------------------------------------------
[109]147%read the current calibration points
[2]148Coord_cell=get(handles.ListCoord,'String');
149Object=read_geometry_calib(Coord_cell);
[109]150Coord=Object.Coord;
151% apply the calibration, whose type is selected in  handles.calib_type
152if ~isempty(Coord)
153    calib_cell=get(handles.calib_type,'String');
154    val=get(handles.calib_type,'Value');
155    GeometryCalib=feval(['calib_' calib_cell{val}],Coord,handles);
156else
157    msgbox_uvmat('ERROR','No calibration points, abort')
158    return
159end
[114]160Z_plane=[];
[109]161if ~isempty(Coord)
[114]162    %check error
[109]163    X=Coord(:,1);
164    Y=Coord(:,2);
165    Z=Coord(:,3);
166    x_ima=Coord(:,4);
167    y_ima=Coord(:,5);
[114]168    [Xpoints,Ypoints]=px_XYZ(GeometryCalib,X,Y,Z);
[109]169    GeometryCalib.ErrorRms(1)=sqrt(mean((Xpoints-x_ima).*(Xpoints-x_ima)));
170    [GeometryCalib.ErrorMax(1),index(1)]=max(abs(Xpoints-x_ima));
171    GeometryCalib.ErrorRms(2)=sqrt(mean((Ypoints-y_ima).*(Ypoints-y_ima)));
172    [GeometryCalib.ErrorMax(2),index(2)]=max(abs(Ypoints-y_ima));
173    [EM,ind_dim]=max(GeometryCalib.ErrorMax);
174    index=index(ind_dim);
[114]175    %set the Z position of the reference plane used for calibration
[109]176    if isequal(max(Z),min(Z))%Z constant
[114]177        Z_plane=Z(1);
[109]178        GeometryCalib.NbSlice=1;
[114]179        GeometryCalib.SliceCoord=[0 0 Z_plane];
[109]180    end
181end
[114]182%set the coordinate unit
[2]183unitlist=get(handles.CoordUnit,'String');
184unit=unitlist{get(handles.CoordUnit,'value')};
185GeometryCalib.CoordUnit=unit;
[114]186%record the points
[109]187GeometryCalib.SourceCalib.PointCoord=Coord;
[114]188display_intrinsic(GeometryCalib,handles)%display calibration intrinsic parameters
[109]189
[114]190% Display extrinsinc parameters (rotation and translation of camera with  respect to the phys coordiantes)
[109]191set(handles.Tx,'String',num2str(GeometryCalib.Tx_Ty_Tz(1),4))
192set(handles.Ty,'String',num2str(GeometryCalib.Tx_Ty_Tz(2),4))
193set(handles.Tz,'String',num2str(GeometryCalib.Tx_Ty_Tz(3),4))
[114]194set(handles.Phi,'String',num2str(GeometryCalib.omc(1),4))
195set(handles.Theta,'String',num2str(GeometryCalib.omc(2),4))
196set(handles.Psi,'String',num2str(GeometryCalib.omc(3),4))
[109]197
[114]198% store the calibration data, by default in the xml file of the currently displayed image
199hhuvmat=guidata(findobj(allchild(0),'Name','uvmat'));%handles of elements in the GUI uvmat
[2]200RootPath='';
201RootFile='';
[121]202if ~isempty(hhuvmat.RootPath)&& ~isempty(hhuvmat.RootFile)
[2]203    testhandle=1;
204    RootPath=get(hhuvmat.RootPath,'String');
205    RootFile=get(hhuvmat.RootFile,'String');
206    filebase=fullfile(RootPath,RootFile);
[114]207    outputfile=[filebase '.xml'];%xml file associated with the currently displayed image
[2]208else
209    question={'save the calibration data and point coordinates in'};
[121]210    def={fullfile(RootPath,'ObjectCalib.xml')};
[2]211    options.Resize='on';
212    answer=inputdlg(question,'save average in a new file',1,def,options);
213    outputfile=answer{1};
214end
[84]215answer=msgbox_uvmat('INPUT_Y-N',{[outputfile ' updated with calibration data'];...
[2]216    ['Error rms (along x,y)=' num2str(GeometryCalib.ErrorRms) ' pixels'];...
[84]217    ['Error max (along x,y)=' num2str(GeometryCalib.ErrorMax) ' pixels']});
[128]218
219%% record the calibration parameters and display the current image of uvmat in the new phys coordinates
[121]220if strcmp(answer,'Yes')
221    if strcmp(calib_cell{val}(1:2),'3D')%set the plane position for 3D (projection) calibration
222        answer_1=msgbox_uvmat('INPUT_TXT',' Z= ',num2str(Z_plane));
223        if strcmp(answer_1,'Cancel')
224            Z_plane=0; %default
225        else
[128]226            Z_plane=str2double(answer_1);
[121]227        end
228        GeometryCalib.NbSlice=1;
229        GeometryCalib.SliceCoord=[0 0 Z_plane];
230    end
[114]231    errormsg=update_imadoc(GeometryCalib,outputfile);% introduce the calibration data in the xml file
232    if ~strcmp(errormsg,'')
233        msgbox_uvmat('ERROR',errormsg);
234    end
[128]235   
[84]236    %display image with new calibration in the currently opened uvmat interface
237    hhh=findobj(hhuvmat.axes3,'Tag','calib_marker');% delete calib points and markers
238    if ~isempty(hhh)
[109]239        delete(hhh);     
[84]240    end
241    hhh=findobj(hhuvmat.axes3,'Tag','calib_points');
242    if ~isempty(hhh)
243        delete(hhh);
244    end
245    set(hhuvmat.FixedLimits,'Value',0)% put FixedLimits option to 'off'
246    set(hhuvmat.FixedLimits,'BackgroundColor',[0.7 0.7 0.7])
[116]247    UserData=get(handles.geometry_calib,'UserData');
248    UserData.XmlInputFile=outputfile;%save the current xml file name
249    set(handles.geometry_calib,'UserData',UserData)
[114]250    uvmat('RootPath_Callback',hObject,eventdata,hhuvmat); %file input with xml reading  in uvmat, show the image in phys coordinates
251    MenuPlot_Callback(hObject, eventdata, handles)
252    set(handles.ListCoord,'Value',index)% indicate in the list the point with max deviation (possible mistake)
253    ListCoord_Callback(hObject, eventdata, handles)
[84]254    figure(handles.geometry_calib)
[2]255end
[69]256
257%------------------------------------------------------------------
[2]258% --- Executes on button press in calibrate_lin.
[128]259
[2]260function REPLICATE_Callback(hObject, eventdata, handles)
[60]261%------------------------------------------------------------------------
[2]262
[128]263%% Apply calibration
264calib_cell=get(handles.calib_type,'String'); %#ok<NASGU>
265val=get(handles.calib_type,'Value'); %#ok<NASGU>
266
[114]267%read the current calibration points
268Coord_cell=get(handles.ListCoord,'String');
269Object=read_geometry_calib(Coord_cell);
270Coord=Object.Coord;
[2]271
[114]272% apply the calibration, whose type is selected in  handles.calib_type
273if ~isempty(Coord)
274    calib_cell=get(handles.calib_type,'String');
275    val=get(handles.calib_type,'Value');
276    GeometryCalib=feval(['calib_' calib_cell{val}],Coord,handles);
277else
278    msgbox_uvmat('ERROR','No calibration points, abort')
279    return
280end
281
282if ~isempty(Coord)
283    %check error
284    X=Coord(:,1);
285    Y=Coord(:,2);
286    Z=Coord(:,3);
287    x_ima=Coord(:,4);
288    y_ima=Coord(:,5);
289    [Xpoints,Ypoints]=px_XYZ(GeometryCalib,X,Y,Z);
290    GeometryCalib.ErrorRms(1)=sqrt(mean((Xpoints-x_ima).*(Xpoints-x_ima)));
291    [GeometryCalib.ErrorMax(1),index(1)]=max(abs(Xpoints-x_ima));
292    GeometryCalib.ErrorRms(2)=sqrt(mean((Ypoints-y_ima).*(Ypoints-y_ima)));
293    [GeometryCalib.ErrorMax(2),index(2)]=max(abs(Ypoints-y_ima));
294    [EM,ind_dim]=max(GeometryCalib.ErrorMax);
[128]295%     index=index(ind_dim);
[114]296    %set the Z position of the reference plane used for calibration
297    Z_plane=[];
298    if isequal(max(Z),min(Z))
299        Z_plane=Z(1);
300    end
301    answer_1=msgbox_uvmat('INPUT_TXT',' Z= ',num2str(Z_plane));
[128]302    Z_plane=str2double(answer_1);
[114]303    GeometryCalib.NbSlice=1;
304    GeometryCalib.SliceCoord=[0 0 Z_plane];
305    %set the coordinate unit
306    unitlist=get(handles.CoordUnit,'String');
307    unit=unitlist{get(handles.CoordUnit,'value')};
308    GeometryCalib.CoordUnit=unit;
309    %record the points
310    GeometryCalib.SourceCalib.PointCoord=Coord;
311end
[128]312
313%% display calibration paprameters
[114]314display_intrinsic(GeometryCalib,handles)%display calibration intrinsic parameters
315
316% Display extrinsinc parameters (rotation and translation of camera with  respect to the phys coordiantes)
317set(handles.Tx,'String',num2str(GeometryCalib.Tx_Ty_Tz(1),4))
318set(handles.Ty,'String',num2str(GeometryCalib.Tx_Ty_Tz(2),4))
319set(handles.Tz,'String',num2str(GeometryCalib.Tx_Ty_Tz(3),4))
320set(handles.Phi,'String',num2str(GeometryCalib.omc(1),4))
321set(handles.Theta,'String',num2str(GeometryCalib.omc(2),4))
322set(handles.Psi,'String',num2str(GeometryCalib.omc(3),4))
323
[128]324%% open the GUI dataview
[2]325h_dataview=findobj(allchild(0),'name','dataview');
[12]326if ~isempty(h_dataview)
327    delete(h_dataview)
[2]328end
[61]329CalibData=get(handles.geometry_calib,'UserData');%read the calibration image source on the interface userdata
[128]330InputFile='';
[114]331if isfield(CalibData,'XmlInputFile')
[128]332    InputDir=fileparts(CalibData.XmlInputFile);
333    [InputDir,DirName]=fileparts(InputDir);
[2]334end
[12]335SubCampaignTest='n'; %default
[128]336testup=0;
337if isfield(CalibData,'SubCampaign')
338    SubCampaignTest='y';
339    dir_ref=CalibData.SubCampaign;
340    testup=1;
341elseif isfield(CalibData,'Campaign')
342    dir_ref=CalibData.Campaign;
343    testup=1;
[2]344end
[128]345while testup
346    [InputDir,DirName]=fileparts(InputDir);
347    if strcmp(DirName,dir_ref)
348        break
[2]349    end
350end
[128]351InputDir=fullfile(InputDir,DirName);
352answer=msgbox_uvmat('INPUT_TXT','Campaign ?',InputDir);
353if strcmp(answer,'Cancel')
354    return
[2]355end
356
[128]357dataview(answer,SubCampaignTest,GeometryCalib);
358       
359%     if isfield(Heading,'Device') && isequal([filename ext],Heading.Device)
360%         [XmlInput,filename,ext]=fileparts(XmlInput);
361%         Device=Heading.Device;
362%     end
363%     if isfield(Heading,'Experiment') && isequal([filename ext],Heading.Experiment)
364%         [PP,filename,ext]=fileparts(XmlInput);
365%     end
366%     testinput=0;
367%     if isfield(Heading,'SubCampaign') && isequal([filename ext],Heading.SubCampaign)
368%         SubCampaignTest='y';
369%         testinput=1;
370%     elseif isfield(Heading,'Campaign') && isequal([filename ext],Heading.Campaign)
371%         testinput=1;
372% %     end
373% end
374% if ~testinput
375%     filename='PROJETS';%default
376%     if isfield(CalibData,'XmlInputFile')
377%          [pp,filename]=fileparts(CalibData.XmlInputFile);
378%     end
379%     while ~isequal(filename,'PROJETS') && numel(filename)>1
380%         filename_1=filename;
381%         pp_1=pp;
382%         [pp,filename]=fileparts(pp);
383%     end
384%     XmlInput=fullfile(pp_1,filename_1);
385%     testinput=1;
386% end
387% if testinput
388%     outcome=dataview(XmlInput,SubCampaignTest,GeometryCalib);
389% end
390
[60]391%------------------------------------------------------------------------
[2]392% determine the parameters for a calibration by an affine function (rescaling and offset, no rotation)
[109]393function GeometryCalib=calib_rescale(Coord,handles)
[60]394%------------------------------------------------------------------------
[2]395X=Coord(:,1);
[109]396Y=Coord(:,2);% Z not used
[2]397x_ima=Coord(:,4);
398y_ima=Coord(:,5);
399[px,sx]=polyfit(X,x_ima,1);
400[py,sy]=polyfit(Y,y_ima,1);
401T_x=px(2);
402T_y=py(2);
403GeometryCalib.CalibrationType='rescale';
[121]404GeometryCalib.fx_fy=[px(1) py(1)];%.fx_fy corresponds to pxcm along x and y
[2]405GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
[114]406GeometryCalib.Tx_Ty_Tz=[px(2)/px(1) py(2)/py(1) 1];
407%GeometryCalib.R=[1,0,0;0,1,0;0,0,0];
408GeometryCalib.omc=[0 0 0];
[2]409
[60]410%------------------------------------------------------------------------
[2]411% determine the parameters for a calibration by a linear transform matrix (rescale and rotation)
[114]412function GeometryCalib=calib_linear(Coord,handles) %TO UPDATE
[60]413%------------------------------------------------------------------------
[2]414X=Coord(:,1);
[109]415Y=Coord(:,2);% Z not used
[2]416x_ima=Coord(:,4);
417y_ima=Coord(:,5);
418XY_mat=[ones(size(X)) X Y];
419a_X1=XY_mat\x_ima; %transformation matrix for X
[109]420% x1=XY_mat*a_X1;%reconstruction
421% err_X1=max(abs(x1-x_ima));%error
[2]422a_Y1=XY_mat\y_ima;%transformation matrix for X
[109]423% y1=XY_mat*a_Y1;
424% err_Y1=max(abs(y1-y_ima));%error
[121]425% R=[a_X1(2),a_X1(3),0;a_Y1(2),a_Y1(3),0;0,0,1];
426R=[a_X1(2),a_X1(3);a_Y1(2),a_Y1(3)];
[116]427norm=abs(det(R));
[2]428GeometryCalib.CalibrationType='linear';
[121]429GeometryCalib.fx_fy(1)=sqrt((a_X1(2)/a_Y1(3))*norm);
430GeometryCalib.fx_fy(2)=(a_Y1(3)/a_X1(2))*GeometryCalib.fx_fy(1);
[2]431GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
[116]432GeometryCalib.Tx_Ty_Tz=[a_X1(1) a_Y1(1) 1];
[121]433R(1,:)=R(1,:)/GeometryCalib.fx_fy(1);
434R(2,:)=R(2,:)/GeometryCalib.fx_fy(2);
435R=[R;[0 0]];
436GeometryCalib.R=[R [0;0;1]];
[128]437GeometryCalib.omc=(180/pi)*[acos(GeometryCalib.R(1,1)) 0 0];
[109]438%------------------------------------------------------------------------
439% determine the tsai parameters for a view normal to the grid plane
[114]440% NOT USED
[109]441function GeometryCalib=calib_normal(Coord,handles)
442%------------------------------------------------------------------------
443Calib.f1=str2num(get(handles.fx,'String'));
444Calib.f2=str2num(get(handles.fy,'String'));
[114]445Calib.k=str2num(get(handles.kc,'String'));
[109]446Calib.Cx=str2num(get(handles.Cx,'String'));
447Calib.Cy=str2num(get(handles.Cy,'String'));
448%default
449if isempty(Calib.f1)
450    Calib.f1=25/0.012;
451end
452if isempty(Calib.f2)
453    Calib.f2=25/0.012;
454end
455if isempty(Calib.k)
456    Calib.k=0;
457end
458if isempty(Calib.Cx)||isempty(Calib.Cy)
459    huvmat=findobj(allchild(0),'Tag','uvmat');
460    hhuvmat=guidata(huvmat);
461    Calib.Cx=str2num(get(hhuvmat.npx,'String'))/2;
462    Calib.Cx=str2num(get(hhuvmat.npy,'String'))/2;
463end   
464%tsai parameters
465Calib.dpx=0.012;%arbitrary
466Calib.dpy=0.012;
467Calib.sx=Calib.f1*Calib.dpx/(Calib.f2*Calib.dpy);
468Calib.f=Calib.f2*Calib.dpy;
469Calib.kappa1=Calib.k/(Calib.f*Calib.f);
[2]470
[109]471%initial guess
472X=Coord(:,1);
473Y=Coord(:,2);
474Zmean=mean(Coord(:,3));
475x_ima=Coord(:,4)-Calib.Cx;
476y_ima=Coord(:,5)-Calib.Cy;
477XY_mat=[ones(size(X)) X Y];
478a_X1=XY_mat\x_ima; %transformation matrix for X
479a_Y1=XY_mat\y_ima;%transformation matrix for Y
480R=[a_X1(2),a_X1(3),0;a_Y1(2),a_Y1(3),0;0,0,-1];% rotation+ z axis reversal (upward)
481norm=sqrt(det(-R));
482calib_param(1)=0;% quadratic distortion
483calib_param(2)=a_X1(1);
484calib_param(3)=a_Y1(1);
485calib_param(4)=Calib.f/(norm*Calib.dpx)-R(3,3)*Zmean;
[121]486calib_param(5)=angle(a_X1(2)+1i*a_X1(3));
[109]487display(['initial guess=' num2str(calib_param)])
488
489%optimise the parameters: minimisation of error
[121]490calib_param = fminsearch(@(calib_param) error_calib(calib_param,Calib,Coord),calib_param);
[109]491
492GeometryCalib.CalibrationType='tsai_normal';
493GeometryCalib.focal=Calib.f;
494GeometryCalib.dpx_dpy=[Calib.dpx Calib.dpy];
495GeometryCalib.Cx_Cy=[Calib.Cx Calib.Cy];
496GeometryCalib.sx=Calib.sx;
497GeometryCalib.kappa1=calib_param(1);
498GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
499GeometryCalib.Tx_Ty_Tz=[calib_param(2) calib_param(3) calib_param(4)];
500alpha=calib_param(5);
501GeometryCalib.R=[cos(alpha) sin(alpha) 0;-sin(alpha) cos(alpha) 0;0 0 -1];
502
[60]503%------------------------------------------------------------------------
[109]504function GeometryCalib=calib_3D_linear(Coord,handles)
[69]505%------------------------------------------------------------------
506path_uvmat=which('uvmat');% check the path detected for source file uvmat
507path_UVMAT=fileparts(path_uvmat); %path to UVMAT
[78]508huvmat=findobj(allchild(0),'Tag','uvmat');
509hhuvmat=guidata(huvmat);
[121]510coord_files=get(handles.coord_files,'String');
511if ischar(coord_files)
512    coord_files={coord_files};
513end
514if isempty(coord_files{1}) || isequal(coord_files,{''})
515    coord_files={};
516end
517%retrieve the calibration points stored in the files listed in the popup list coord_files
518x_1=Coord(:,4:5)';%px coordinates of the ref points
[109]519nx=str2num(get(hhuvmat.npx,'String'));
520ny=str2num(get(hhuvmat.npy,'String'));
[121]521x_1(2,:)=ny-x_1(2,:);%reverse the y image coordinates
522X_1=Coord(:,1:3)';%phys coordinates of the ref points
523n_ima=numel(coord_files)+1;
524if ~isempty(coord_files)
525    msgbox_uvmat('CONFIRMATION',['The xy coordinates of the calibration points in ' num2str(n_ima) ' planes will be used'])
526    for ifile=1:numel(coord_files)
527    t=xmltree(coord_files{ifile});
528    s=convert(t);%convert to matlab structure
529        if isfield(s,'GeometryCalib')
530            if isfield(s.GeometryCalib,'SourceCalib')
531                if isfield(s.GeometryCalib.SourceCalib,'PointCoord')
532                PointCoord=s.GeometryCalib.SourceCalib.PointCoord;
533                Coord_file=zeros(length(PointCoord),5);%default
534                for i=1:length(PointCoord)
535                    line=str2num(PointCoord{i});
536                    Coord_file(i,4:5)=line(4:5);%px x
537                    Coord_file(i,1:3)=line(1:3);%phys x
538                end
539                eval(['x_' num2str(ifile+1) '=Coord_file(:,4:5)'';']);
540                eval(['x_' num2str(ifile+1) '(2,:)=ny-x_' num2str(ifile+1) '(2,:);' ]);
541                eval(['X_' num2str(ifile+1) '=Coord_file(:,1:3)'';']);
542                end
543            end
544        end
545    end
546end
547n_ima=numel(coord_files)+1;
[109]548est_dist=[0;0;0;0;0];
549est_aspect_ratio=0;
550est_fc=[1;1];
551%fc=[25;25]/0.012;
552center_optim=0;
553run(fullfile(path_UVMAT,'toolbox_calib','go_calib_optim'));
554GeometryCalib.CalibrationType='3D_linear';
[121]555GeometryCalib.fx_fy=fc';
556%GeometryCalib.focal=fc(2);
557%GeometryCalib.dpx_dpy=[1 1];
[109]558GeometryCalib.Cx_Cy=cc';
[121]559%GeometryCalib.sx=fc(1)/fc(2);
560GeometryCalib.kc=kc(1);
561%GeometryCalib.kappa1=-kc(1)/fc(2)^2;
[109]562GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
563GeometryCalib.Tx_Ty_Tz=Tc_1';
564GeometryCalib.R=Rc_1;
[121]565GeometryCalib.R(2,1:3)=-GeometryCalib.R(2,1:3);%inversion of the y image coordinate
566GeometryCalib.Tx_Ty_Tz(2)=-GeometryCalib.Tx_Ty_Tz(2);%inversion of the y image coordinate
567GeometryCalib.Cx_Cy(2)=ny-GeometryCalib.Cx_Cy(2);%inversion of the y image coordinate
568GeometryCalib.omc=(180/pi)*omc_1;%angles in degrees
569GeometryCalib.ErrorRMS=[];
570GeometryCalib.ErrorMax=[];
[109]571
572%------------------------------------------------------------------------
573function GeometryCalib=calib_3D_quadr(Coord,handles)
574%------------------------------------------------------------------
575
576path_uvmat=which('uvmat');% check the path detected for source file uvmat
577path_UVMAT=fileparts(path_uvmat); %path to UVMAT
578huvmat=findobj(allchild(0),'Tag','uvmat');
579hhuvmat=guidata(huvmat);
580% check_cond=0;
581coord_files=get(handles.coord_files,'String');
582if ischar(coord_files)
583    coord_files={coord_files};
584end
585if isempty(coord_files{1}) || isequal(coord_files,{''})
586    coord_files={};
587end
588
589%retrieve the calibration points stored in the files listed in the popup list coord_files
[114]590x_1=Coord(:,4:5)';%px coordinates of the ref points
591nx=str2num(get(hhuvmat.npx,'String'));
592ny=str2num(get(hhuvmat.npy,'String'));
593x_1(2,:)=ny-x_1(2,:);%reverse the y image coordinates
594X_1=Coord(:,1:3)';%phys coordinates of the ref points
[109]595n_ima=numel(coord_files)+1;
596if ~isempty(coord_files)
597    msgbox_uvmat('CONFIRMATION',['The xy coordinates of the calibration points in ' num2str(n_ima) ' planes will be used'])
598    for ifile=1:numel(coord_files)
599    t=xmltree(coord_files{ifile});
600    s=convert(t);%convert to matlab structure
601        if isfield(s,'GeometryCalib')
602            if isfield(s.GeometryCalib,'SourceCalib')
603                if isfield(s.GeometryCalib.SourceCalib,'PointCoord')
604                PointCoord=s.GeometryCalib.SourceCalib.PointCoord;
[121]605                Coord_file=zeros(length(PointCoord),5);%default
[109]606                for i=1:length(PointCoord)
607                    line=str2num(PointCoord{i});
608                    Coord_file(i,4:5)=line(4:5);%px x
609                    Coord_file(i,1:3)=line(1:3);%phys x
610                end
611                eval(['x_' num2str(ifile+1) '=Coord_file(:,4:5)'';']);
[114]612                eval(['x_' num2str(ifile+1) '(2,:)=ny-x_' num2str(ifile+1) '(2,:);' ]);
[109]613                eval(['X_' num2str(ifile+1) '=Coord_file(:,1:3)'';']);
614                end
615            end
616        end
617    end
618end
619n_ima=numel(coord_files)+1;
[108]620est_dist=[1;0;0;0;0];
[109]621est_aspect_ratio=1;
622%est_fc=[0;0];
623%fc=[25;25]/0.012;
624center_optim=0;
[83]625run(fullfile(path_UVMAT,'toolbox_calib','go_calib_optim'));
[69]626
[109]627GeometryCalib.CalibrationType='3D_quadr';
[114]628GeometryCalib.fx_fy=fc';
629%GeometryCalib.focal=fc(2);
630%GeometryCalib.dpx_dpy=[1 1];
[69]631GeometryCalib.Cx_Cy=cc';
[114]632%GeometryCalib.sx=fc(1)/fc(2);
633GeometryCalib.kc=kc(1);
634%GeometryCalib.kappa1=-kc(1)/fc(2)^2;
[69]635GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
636GeometryCalib.Tx_Ty_Tz=Tc_1';
637GeometryCalib.R=Rc_1;
[114]638GeometryCalib.R(2,1:3)=-GeometryCalib.R(2,1:3);%inversion of the y image coordinate
639GeometryCalib.Tx_Ty_Tz(2)=-GeometryCalib.Tx_Ty_Tz(2);%inversion of the y image coordinate
640GeometryCalib.Cx_Cy(2)=ny-GeometryCalib.Cx_Cy(2);%inversion of the y image coordinate
641GeometryCalib.omc=(180/pi)*omc_1;%angles in degrees
[109]642GeometryCalib.ErrorRMS=[];
643GeometryCalib.ErrorMax=[];
[69]644
[109]645
[60]646%------------------------------------------------------------------------
[109]647function GeometryCalib=calib_3D_extrinsic(Coord,handles)
648%------------------------------------------------------------------
649path_uvmat=which('geometry_calib');% check the path detected for source file uvmat
650path_UVMAT=fileparts(path_uvmat); %path to UVMAT
[116]651x_1=double(Coord(:,4:5)');%image coordiantes
652X_1=double(Coord(:,1:3)');% phys coordinates
[114]653huvmat=findobj(allchild(0),'Tag','uvmat');
654hhuvmat=guidata(huvmat);
[116]655ny=str2double(get(hhuvmat.npy,'String'));
[114]656x_1(2,:)=ny-x_1(2,:);%reverse the y image coordinates
[109]657n_ima=1;
[114]658GeometryCalib.CalibrationType='3D_extrinsic';
659GeometryCalib.fx_fy(1)=str2num(get(handles.fx,'String'));
660GeometryCalib.fx_fy(2)=str2num(get(handles.fy,'String'));
661GeometryCalib.Cx_Cy(1)=str2num(get(handles.Cx,'String'));
662GeometryCalib.Cx_Cy(2)=str2num(get(handles.Cy,'String'));
663GeometryCalib.kc=str2num(get(handles.kc,'String'));
[109]664fct_path=fullfile(path_UVMAT,'toolbox_calib');
665addpath(fct_path)
[116]666GeometryCalib.Cx_Cy(2)=ny-GeometryCalib.Cx_Cy(2);%reverse Cx_Cy(2) for calibration (inversion of px ordinate)
[109]667% [omc1,Tc1,Rc1,H,x,ex,JJ] = compute_extrinsic(x_1,X_1,...
668%     [Calib.f Calib.f*Calib.sx]',...
669%     [Calib.Cx Calib.Cy]',...
670%     [-Calib.kappa1*Calib.f^2 0 0 0 0]);
[114]671[omc,Tc1,Rc1,H,x,ex,JJ] = compute_extrinsic(x_1,X_1,...
[116]672    (GeometryCalib.fx_fy)',GeometryCalib.Cx_Cy',[GeometryCalib.kc 0 0 0 0]);
[109]673rmpath(fct_path);
674GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
675GeometryCalib.Tx_Ty_Tz=Tc1';
676%inversion of z axis
677GeometryCalib.R=Rc1;
[114]678GeometryCalib.R(2,1:3)=-GeometryCalib.R(2,1:3);%inversion of the y image coordinate
679GeometryCalib.Tx_Ty_Tz(2)=-GeometryCalib.Tx_Ty_Tz(2);%inversion of the y image coordinate
[116]680GeometryCalib.Cx_Cy(2)=ny-GeometryCalib.Cx_Cy(2);%inversion of the y image coordinate
[114]681GeometryCalib.omc=(180/pi)*omc';
[109]682%GeometryCalib.R(3,1:3)=-GeometryCalib.R(3,1:3);%inversion for z upward
683
684
[116]685
[109]686%------------------------------------------------------------------------
687%function GeometryCalib=calib_tsai_heikkila(Coord)
688% TEST: NOT IMPLEMENTED
689%------------------------------------------------------------------
690% path_uvmat=which('uvmat');% check the path detected for source file uvmat
691% path_UVMAT=fileparts(path_uvmat); %path to UVMAT
692% path_calib=fullfile(path_UVMAT,'toolbox_calib_heikkila');
693% addpath(path_calib)
694% npoints=size(Coord,1);
695% Coord(:,1:3)=10*Coord(:,1:3);
696% Coord=[Coord zeros(npoints,2) -ones(npoints,1)];
697% [par,pos,iter,res,er,C]=cacal('dalsa',Coord);
698% GeometryCalib.CalibrationType='tsai';
699% GeometryCalib.focal=par(2);
700
701
702%--------------------------------------------------------------------------
[128]703function GeometryCalib=calib_tsai(Coord,handles)% OBSOLETE: old version using gauthier's bianry ccal_fo
[114]704% NOT USED
[109]705%------------------------------------------------------------------------
[2]706%TSAI
707path_uvmat=which('uvmat');% check the path detected for source file uvmat
708path_UVMAT=fileparts(path_uvmat); %path to UVMAT
[109]709xmlfile=fullfile(path_UVMAT,'PARAM.xml');%name of the file containing names of binary executables
[54]710if exist(xmlfile,'file')
[109]711    t=xmltree(xmlfile);% read the (xml) file containing names of binary executables
712    sparam=convert(t);% convert to matlab structure
[54]713end
[71]714if ~isfield(sparam,'GeometryCalibBin')
715    msgbox_uvmat('ERROR',['calibration program <GeometryCalibBin> undefined in parameter file ' xmlfile])
[2]716    return
717end
[71]718Tsai_exe=sparam.GeometryCalibBin;
[54]719if ~exist(Tsai_exe,'file')%the binary is defined in /bin, default setting
720     Tsai_exe=fullfile(path_UVMAT,Tsai_exe);
721end
[2]722if ~exist(Tsai_exe,'file')
[71]723    msgbox_uvmat('ERROR',['calibration program ' sparam.GeometryCalibBin ' defined in PARAM.xml does not exist'])
[2]724    return
725end
726
727textcoord=num2str(Coord,4);
728dlmwrite('t.txt',textcoord,''); 
[109]729% ['!' Tsai_exe ' -fx 0 -fy t.txt']
730eval(['!' Tsai_exe ' -f t.txt > tsaicalib.log']);
[2]731if ~exist('calib.dat','file')
[42]732    msgbox_uvmat('ERROR','no output from calibration program Tsai_exe: possibly too few points')
[2]733end
734calibdat=dlmread('calib.dat');
[60]735delete('calib.dat')
[109]736%delete('t.txt')
[2]737GeometryCalib.CalibrationType='tsai';
738GeometryCalib.focal=calibdat(10);
739GeometryCalib.dpx_dpy=[calibdat(5) calibdat(6)];
740GeometryCalib.Cx_Cy=[calibdat(7) calibdat(8)];
741GeometryCalib.sx=calibdat(9);
742GeometryCalib.kappa1=calibdat(11);
743GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
744GeometryCalib.Tx_Ty_Tz=[calibdat(12) calibdat(13) calibdat(14)];
[121]745Rx_Ry_Rz=calibdat(15:17);
[2]746sa = sin(Rx_Ry_Rz(1)) ;
747ca=cos(Rx_Ry_Rz(1));
748sb=sin(Rx_Ry_Rz(2));
749cb =cos(Rx_Ry_Rz(2));
750sg =sin(Rx_Ry_Rz(3));
751cg =cos(Rx_Ry_Rz(3));
752r1 = cb * cg;
753r2 = cg * sa * sb - ca * sg;
754r3 = sa * sg + ca * cg * sb;
755r4 = cb * sg;
756r5 = sa * sb * sg + ca * cg;
757r6 = ca * sb * sg - cg * sa;
758r7 = -sb;
759r8 = cb * sa;
760r9 = ca * cb;
761%EN DEDUIRE MATRICE R ??
762GeometryCalib.R=[r1,r2,r3;r4,r5,r6;r7,r8,r9];
763
[60]764%------------------------------------------------------------------------
[109]765% --- determine the rms of calibration error
766function ErrorRms=error_calib(calib_param,Calib,Coord)
767%calib_param: vector of free calibration parameters (to optimise)
768%Calib: structure of the given calibration parameters
769%Coord: list of phys coordinates (columns 1-3, and pixel coordinates (columns 4-5)
770Calib.f=25;
771Calib.dpx=0.012;
772Calib.dpy=0.012;
773Calib.sx=1;
774Calib.Cx=512;
775Calib.Cy=512;
776Calib.kappa1=calib_param(1);
777Calib.Tx=calib_param(2);
778Calib.Ty=calib_param(3);
779Calib.Tz=calib_param(4);
780alpha=calib_param(5);
781Calib.R=[cos(alpha) sin(alpha) 0;-sin(alpha) cos(alpha) 0;0 0 -1];
[2]782
[109]783X=Coord(:,1);
784Y=Coord(:,2);
785Z=Coord(:,3);
786x_ima=Coord(:,4);
787y_ima=Coord(:,5);
788[Xpoints,Ypoints]=px_XYZ(Calib,X,Y,Z);
789ErrorRms(1)=sqrt(mean((Xpoints-x_ima).*(Xpoints-x_ima)));
790ErrorRms(2)=sqrt(mean((Ypoints-y_ima).*(Ypoints-y_ima)));
791ErrorRms=mean(ErrorRms);
792
[60]793%------------------------------------------------------------------------
[2]794function XImage_Callback(hObject, eventdata, handles)
[60]795%------------------------------------------------------------------------
[2]796update_list(hObject, eventdata,handles)
797
[60]798%------------------------------------------------------------------------
[2]799function YImage_Callback(hObject, eventdata, handles)
[60]800%------------------------------------------------------------------------
[2]801update_list(hObject, eventdata,handles)
802
[109]803%------------------------------------------------------------------------
804% --- Executes on button press in STORE.
805function STORE_Callback(hObject, eventdata, handles)
806Coord_cell=get(handles.ListCoord,'String');
807Object=read_geometry_calib(Coord_cell);
808unitlist=get(handles.CoordUnit,'String');
809unit=unitlist{get(handles.CoordUnit,'value')};
810GeometryCalib.CoordUnit=unit;
811GeometryCalib.SourceCalib.PointCoord=Object.Coord;
812huvmat=findobj(allchild(0),'Name','uvmat');
813hhuvmat=guidata(huvmat);%handles of elements in the GUI uvmat
[128]814% RootPath='';
815% RootFile='';
[121]816if ~isempty(hhuvmat.RootPath)&& ~isempty(hhuvmat.RootFile)
[109]817    testhandle=1;
818    RootPath=get(hhuvmat.RootPath,'String');
819    RootFile=get(hhuvmat.RootFile,'String');
820    filebase=fullfile(RootPath,RootFile);
821    while exist([filebase '.xml'],'file')
822        filebase=[filebase '~'];
823    end
824    outputfile=[filebase '.xml'];
[114]825    errormsg=update_imadoc(GeometryCalib,outputfile);
826    if ~strcmp(errormsg,'')
827        msgbox_uvmat('ERROR',errormsg);
828    end
[109]829    listfile=get(handles.coord_files,'string');
830    if isequal(listfile,{''})
831        listfile={outputfile};
832    else
[128]833        listfile=[listfile;{outputfile}];%update the list of coord files
[109]834    end
835    set(handles.coord_files,'string',listfile);
836end
837set(handles.ListCoord,'Value',1)% refresh the display of coordinates
838set(handles.ListCoord,'String',{'......'})
839
[114]840% --------------------------------------------------------------------
841% --- Executes on button press in CLEAR_PTS: clear the list of calibration points
842function CLEAR_PTS_Callback(hObject, eventdata, handles)
843% --------------------------------------------------------------------
844set(handles.ListCoord,'Value',1)% refresh the display of coordinates
845set(handles.ListCoord,'String',{'......'})
846MenuPlot_Callback(hObject, eventdata, handles)
847
[109]848%------------------------------------------------------------------------
849% --- Executes on button press in CLEAR.
850function CLEAR_Callback(hObject, eventdata, handles)
851%------------------------------------------------------------------------
852set(handles.coord_files,'Value',1)
853set(handles.coord_files,'String',{''})
854
855%------------------------------------------------------------------------
[2]856function XObject_Callback(hObject, eventdata, handles)
[109]857%------------------------------------------------------------------------
[2]858update_list(hObject, eventdata,handles)
859
[109]860%------------------------------------------------------------------------
[2]861function YObject_Callback(hObject, eventdata, handles)
[109]862%------------------------------------------------------------------------
[2]863update_list(hObject, eventdata,handles)
864
[109]865%------------------------------------------------------------------------
[2]866function ZObject_Callback(hObject, eventdata, handles)
[109]867%------------------------------------------------------------------------
[2]868update_list(hObject, eventdata,handles)
869
[60]870%------------------------------------------------------------------------
[2]871function update_list(hObject, eventdata, handles)
[60]872%------------------------------------------------------------------------
[2]873str4=get(handles.XImage,'String');
874str5=get(handles.YImage,'String');
875str1=get(handles.XObject,'String');
876tt=double(str1);
877str2=get(handles.YObject,'String');
878str3=get(handles.ZObject,'String');
[121]879if ~isempty(str1) && ~isequal(double(str1),32) && (isempty(str3)||isequal(double(str3),32))
[2]880    str3='0';%put z to 0 by default
881end
882strline=[str1 '    |    ' str2 '    |    ' str3 '    |    ' str4 '    |    ' str5];
883Coord=get(handles.ListCoord,'String');
884val=get(handles.ListCoord,'Value');
885Coord{val}=strline;
886set(handles.ListCoord,'String',Coord)
[60]887%update the plot
888ListCoord_Callback(hObject, eventdata, handles)
[67]889MenuPlot_Callback(hObject, eventdata, handles)
[71]890
[60]891%------------------------------------------------------------------------
[2]892% --- Executes on selection change in ListCoord.
893function ListCoord_Callback(hObject, eventdata, handles)
[60]894%------------------------------------------------------------------------
[71]895huvmat=findobj(allchild(0),'Name','uvmat');%find the current uvmat interface handle
896hplot=findobj(huvmat,'Tag','axes3');%main plotting axis of uvmat
897hhh=findobj(hplot,'Tag','calib_marker');
[2]898Coord_cell=get(handles.ListCoord,'String');
899val=get(handles.ListCoord,'Value');
[78]900if numel(val)>1
901    return %no action if several lines have been selected
902end
[71]903coord_str=Coord_cell{val};
[128]904k=findstr(' | ',coord_str);
[71]905if isempty(k)%last line '.....' selected
906    if ~isempty(hhh)
907        delete(hhh)%delete the circle marker
[2]908    end
[71]909    return
910end
911%fill the edit boxex
912set(handles.XObject,'String',coord_str(1:k(1)-5))
913set(handles.YObject,'String',coord_str(k(1)+5:k(2)-5))
914set(handles.ZObject,'String',coord_str(k(2)+5:k(3)-5))
915set(handles.XImage,'String',coord_str(k(3)+5:k(4)-5))
916set(handles.YImage,'String',coord_str(k(4)+5:end))
917h_menu_coord=findobj(huvmat,'Tag','transform_fct');
918menu=get(h_menu_coord,'String');
919choice=get(h_menu_coord,'Value');
920if iscell(menu)
921    option=menu{choice};
922else
923    option='px'; %default
924end
925if isequal(option,'phys')
926    XCoord=str2num(coord_str(1:k(1)-5));
927    YCoord=str2num(coord_str(k(1)+5:k(2)-5));
928elseif isequal(option,'px')|| isequal(option,'')
929    XCoord=str2num(coord_str(k(3)+5:k(4)-5));
930    YCoord=str2num(coord_str(k(4)+5:end));
931else
932    msgbox_uvmat('ERROR','the choice in menu_coord of uvmat must be px or phys ')
933end
934if isempty(XCoord)||isempty(YCoord)
935     if ~isempty(hhh)
936        delete(hhh)%delete the circle marker
[2]937    end
[71]938    return
[2]939end
[71]940xlim=get(hplot,'XLim');
941ylim=get(hplot,'YLim');
942ind_range=max(abs(xlim(2)-xlim(1)),abs(ylim(end)-ylim(1)))/20;%defines the size of the circle marker
943if isempty(hhh)
944    axes(hplot)
945    rectangle('Curvature',[1 1],...
946              'Position',[XCoord-ind_range/2 YCoord-ind_range/2 ind_range ind_range],'EdgeColor','m',...
947              'LineStyle','-','Tag','calib_marker');
948else
949    set(hhh,'Position',[XCoord-ind_range/2 YCoord-ind_range/2 ind_range ind_range])
950end
[2]951
[60]952%------------------------------------------------------------------------
[2]953% --- Executes on selection change in edit_append.
954function edit_append_Callback(hObject, eventdata, handles)
[60]955%------------------------------------------------------------------------
[2]956choice=get(handles.edit_append,'Value');
[78]957if choice
958    set(handles.edit_append,'BackgroundColor',[1 1 0])
959else
960    set(handles.edit_append,'BackgroundColor',[0.7 0.7 0.7])
[2]961end
962   
963function NEW_Callback(hObject, eventdata, handles)
964%A METTRE SOUS UN BOUTON
965huvmat=findobj(allchild(0),'Name','uvmat');
966hchild=get(huvmat,'children');
967hcoord=findobj(hchild,'Tag','menu_coord');
968coordtype=get(hcoord,'Value');
969haxes=findobj(hchild,'Tag','axes3');
970AxeData=get(haxes,'UserData');
971if ~isequal(hcoord,2)
972    set(hcoord,'Value',2)
973    huvmat=uvmat(AxeData);
974    'relancer uvmat';
975end
976if ~isfield(AxeData,'ZoomAxes')
[42]977    msgbox_uvmat('ERROR','first draw a window around a grid marker')
[2]978    return
979end
980XLim=get(AxeData.ZoomAxes,'XLim');
981YLim=get(AxeData.ZoomAxes,'YLim');
982np=size(AxeData.A);
983ind_sub_x=round(XLim);
984ind_sub_y=np(1)-round(YLim);
985Mfiltre=AxeData.A([ind_sub_y(2):ind_sub_y(1)] ,ind_sub_x,:);
986Mfiltre_norm=double(Mfiltre);
987Mfiltre_norm=Mfiltre_norm/sum(sum(Mfiltre_norm));
988Mfiltre_norm=100*(Mfiltre_norm-mean(mean(Mfiltre_norm)));
989Atype=class(AxeData.A);
990Data.NbDim=2;
991Data.A=filter2(Mfiltre_norm,double(AxeData.A));
992Data.A=feval(Atype,Data.A);
993Data.AName='image';
994Data.AX=AxeData.AX;
995Data.AY=AxeData.AY;
996Data.CoordType='px';
997plot_field(Data)
998
[60]999%------------------------------------------------------------------------
[2]1000function MenuPlot_Callback(hObject, eventdata, handles)
[60]1001%------------------------------------------------------------------------
[2]1002huvmat=findobj(allchild(0),'Name','uvmat');%find the current uvmat interface handle
[128]1003%UvData=get(huvmat,'UserData');%Data associated to the current uvmat interface
[2]1004hhuvmat=guidata(huvmat); %handles of GUI elements in uvmat
[128]1005%hplot=findobj(huvmat,'Tag','axes3');%main plotting axis of uvmat
[60]1006h_menu_coord=findobj(huvmat,'Tag','transform_fct');
[2]1007menu=get(h_menu_coord,'String');
1008choice=get(h_menu_coord,'Value');
1009if iscell(menu)
1010    option=menu{choice};
1011else
1012    option='px'; %default
1013end
1014Coord_cell=get(handles.ListCoord,'String');
1015ObjectData=read_geometry_calib(Coord_cell);
1016%ObjectData=read_geometry_calib(handles);%read the interface input parameters defining the object
[71]1017if ~isempty(ObjectData.Coord)
1018    if isequal(option,'phys')
1019        ObjectData.Coord=ObjectData.Coord(:,[1:3]);
1020    elseif isequal(option,'px')||isequal(option,'')
1021        ObjectData.Coord=ObjectData.Coord(:,[4:5]);
1022    else
1023        msgbox_uvmat('ERROR','the choice in menu_coord of uvmat must be '''', px or phys ')
1024    end
[2]1025end
1026axes(hhuvmat.axes3)
1027hh=findobj('Tag','calib_points');
[71]1028if  ~isempty(ObjectData.Coord) && isempty(hh)
[2]1029    hh=line(ObjectData.Coord(:,1),ObjectData.Coord(:,2),'Color','m','Tag','calib_points','LineStyle','.','Marker','+');
[71]1030elseif isempty(ObjectData.Coord)%empty list of points, suppress the plot
1031    delete(hh)
[2]1032else
1033    set(hh,'XData',ObjectData.Coord(:,1))
1034    set(hh,'YData',ObjectData.Coord(:,2))
1035end
[61]1036pause(.1)
1037figure(handles.geometry_calib)
[2]1038
1039% --------------------------------------------------------------------
1040function MenuHelp_Callback(hObject, eventdata, handles)
[116]1041path_to_uvmat=which('uvmat');% check the path of uvmat
[2]1042pathelp=fileparts(path_to_uvmat);
[116]1043helpfile=fullfile(pathelp,'uvmat_doc','uvmat_doc.html');
[36]1044if isempty(dir(helpfile)), msgbox_uvmat('ERROR','Please put the help file uvmat_doc.html in the sub-directory /uvmat_doc of the UVMAT package')
[2]1045else
[36]1046   addpath (fullfile(pathelp,'uvmat_doc'))
[2]1047   web([helpfile '#geometry_calib'])
1048end
1049
[17]1050%------------------------------------------------------------------------
[2]1051function MenuCreateGrid_Callback(hObject, eventdata, handles)
[17]1052%------------------------------------------------------------------------
[36]1053%hcalib=get(handles.calib_type,'parent');%handles of the GUI geometry_calib
[61]1054CalibData=get(handles.geometry_calib,'UserData');
[12]1055Tinput=[];%default
1056if isfield(CalibData,'grid')
1057    Tinput=CalibData.grid;
1058end
[71]1059[T,CalibData.grid]=create_grid(Tinput);%display the GUI create_grid
[61]1060set(handles.geometry_calib,'UserData',CalibData)
[2]1061
[12]1062%grid in phys space
[71]1063Coord=get(handles.ListCoord,'String');
1064val=get(handles.ListCoord,'Value');
1065data=read_geometry_calib(Coord);
1066%nbpoints=size(data.Coord,1); %nbre of calibration points
1067data.Coord(val:val+size(T,1)-1,1:3)=T(end:-1:1,:);%update the existing list of phys coordinates from the GUI create_grid
1068% for i=1:nbpoints
1069%    for j=1:5
1070%           Coord{i,j}=num2str(data.Coord(i,j),4);%display coordiantes with 4 digits
1071%    end
1072% end
1073%update the phys coordinates starting from the selected point (down in the
1074Coord(end,:)=[]; %remove last string '.....'
1075for i=1:size(data.Coord,1)
1076    for j=1:5
[17]1077          Coord{i,j}=num2str(data.Coord(i,j),4);%display coordiantes with 4 digits
1078    end
1079end
1080
1081%size(data.Coord,1)
[128]1082Tabchar=cell2tab(Coord,' | ');
[71]1083Tabchar=[Tabchar ;{'......'}];
[12]1084set(handles.ListCoord,'String',Tabchar)
[2]1085
[71]1086% -----------------------------------------------------------------------
1087% --- automatic grid dectection from local maxima of the images
[60]1088function MenuDetectGrid_Callback(hObject, eventdata, handles)
[71]1089%------------------------------------------------------------------------
[108]1090CalibData=get(handles.geometry_calib,'UserData');%get information stored on the GUI geometry_calib
[60]1091grid_input=[];%default
1092if isfield(CalibData,'grid')
1093    grid_input=CalibData.grid;%retrieve the previously used grid
1094end
[109]1095[T,CalibData.grid]=create_grid(grid_input,'detect_grid');%display the GUI create_grid, read the set of phys coordinates T
1096
[108]1097set(handles.geometry_calib,'UserData',CalibData)%store the phys grid parameters for later use
[60]1098
[108]1099%read the four last point coordinates in pixels
1100Coord_cell=get(handles.ListCoord,'String');%read list of coordinates on geometry_calib
[60]1101data=read_geometry_calib(Coord_cell);
1102nbpoints=size(data.Coord,1); %nbre of calibration points
[62]1103if nbpoints~=4
1104    msgbox_uvmat('ERROR','four points must be selected by the mouse, beginning by the new x axis, to delimitate the phs grid area')
[71]1105    return
[60]1106end
[71]1107corners_X=(data.Coord(end:-1:end-3,4)); %pixel absissa of the four corners
1108corners_Y=(data.Coord(end:-1:end-3,5));
[60]1109
[71]1110%reorder the last two points (the two first in the list) if needed
[121]1111angles=angle((corners_X-corners_X(1))+1i*(corners_Y-corners_Y(1)));
[62]1112if abs(angles(4)-angles(2))>abs(angles(3)-angles(2))
1113      X_end=corners_X(4);
1114      Y_end=corners_Y(4);
1115      corners_X(4)=corners_X(3);
1116      corners_Y(4)=corners_Y(3);
1117      corners_X(3)=X_end;
1118      corners_Y(3)=Y_end;
1119end
1120
[108]1121%read the current image, displayed in the GUI uvmat
[60]1122huvmat=findobj(allchild(0),'Name','uvmat');
1123UvData=get(huvmat,'UserData');
1124A=UvData.Field.A;
1125npxy=size(A);
1126%linear transform on the current image
[108]1127X=[CalibData.grid.x_0 CalibData.grid.x_1 CalibData.grid.x_0 CalibData.grid.x_1]';%corner absissa in the phys coordinates
1128Y=[CalibData.grid.y_0 CalibData.grid.y_0 CalibData.grid.y_1 CalibData.grid.y_1]';%corner ordinates in the phys coordinates
[109]1129
[108]1130%calculate transform matrices:
1131% reference: http://alumni.media.mit.edu/~cwren/interpolator/ by Christopher R. Wren
1132B = [ X Y ones(size(X)) zeros(4,3)        -X.*corners_X -Y.*corners_X ...
1133      zeros(4,3)        X Y ones(size(X)) -X.*corners_Y -Y.*corners_Y ];
1134B = reshape (B', 8 , 8 )';
1135D = [ corners_X , corners_Y ];
1136D = reshape (D', 8 , 1 );
[121]1137%l = inv(B' * B) * B' * D;
1138l = (B' * B)\B' * D;
[108]1139Amat = reshape([l(1:6)' 0 0 1 ],3,3)';
1140C = [l(7:8)' 1];
1141
[114]1142%GeometryCalib.CalibrationType='tsai';
1143%GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
1144% GeometryCalib.f=1;
1145% GeometryCalib.dpx=1;
1146% GeometryCalib.dpy=1;
1147% GeometryCalib.sx=1;
1148% GeometryCalib.Cx=0;
1149% GeometryCalib.Cy=0;
1150% GeometryCalib.kappa1=0;
1151% GeometryCalib.Tx=Amat(1,3);
1152% GeometryCalib.Ty=Amat(2,3);
1153% GeometryCalib.Tz=1;
1154% GeometryCalib.R=[Amat(1,1),Amat(1,2),0;Amat(2,1),Amat(2,2),0;C(1),C(2),0];
1155%
1156% [Amod,Rangx,Rangy]=phys_Ima(A-min(min(A)),GeometryCalib,0);
1157
1158GeometryCalib.fx_fy=[1 1];
1159GeometryCalib.Tx_Ty_Tz=[Amat(1,3) Amat(2,3) 1];
[108]1160GeometryCalib.R=[Amat(1,1),Amat(1,2),0;Amat(2,1),Amat(2,2),0;C(1),C(2),0];
[114]1161path_uvmat=which('uvmat');% check the path detected for source file uvmat
1162path_UVMAT=fileparts(path_uvmat); %path to UVMAT
1163addpath(fullfile(path_UVMAT,'transform_field'))
1164Data.ListVarName={'AY','AX','A'};
1165Data.VarDimName={'AY','AX',{'AY','AX'}};
1166if ndims(A)==3
1167    A=mean(A,3);
1168end
1169Data.A=A-min(min(A));
1170Data.AY=[npxy(1)-0.5 0.5];
1171Data.AX=[0.5 npxy(2)];
1172Data.CoordType='px';
1173Calib.GeometryCalib=GeometryCalib;
[121]1174DataOut=phys(Data,Calib);
[114]1175rmpath(fullfile(path_UVMAT,'transform_field'))
1176Amod=DataOut.A;
1177Rangx=DataOut.AX;
1178Rangy=DataOut.AY;
1179% GeometryCalib.dpx=1;
1180% GeometryCalib.dpy=1;
1181% GeometryCalib.sx=1;
1182% GeometryCalib.Cx=0;
1183% GeometryCalib.Cy=0;
1184% GeometryCalib.kappa1=0;
1185% GeometryCalib.Tx=Amat(1,3);
1186% GeometryCalib.Ty=Amat(2,3);
1187% GeometryCalib.Tz=1;
1188% GeometryCalib.R=[Amat(1,1),Amat(1,2),0;Amat(2,1),Amat(2,2),0;C(1),C(2),0];
1189%
1190% [Amod,Rangx,Rangy]=phys_Ima(A-min(min(A)),GeometryCalib,0);
[108]1191
1192
[60]1193Amod=double(Amod);
[114]1194% figure(12) %display corrected image
[109]1195% Amax=max(max(Amod));
1196% image(Rangx,Rangy,uint8(255*Amod/Amax))
[114]1197
[88]1198Dx=(Rangx(2)-Rangx(1))/(npxy(2)-1); %x mesh in real space
1199Dy=(Rangy(2)-Rangy(1))/(npxy(1)-1); %y mesh in real space
[121]1200ind_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
1201ind_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
[60]1202nbpoints=size(T,1);
1203for ipoint=1:nbpoints
1204    i0=1+round((T(ipoint,1)-Rangx(1))/Dx);%round(Xpx(ipoint));
1205    j0=1+round((T(ipoint,2)-Rangy(1))/Dy);%round(Xpx(ipoint));
[109]1206    j0min=max(j0-ind_range_y,1);
1207    j0max=min(j0+ind_range_y,size(Amod,1));
1208    i0min=max(i0-ind_range_x,1);
1209    i0max=min(i0+ind_range_x,size(Amod,2));
1210    Asub=Amod(j0min:j0max,i0min:i0max);
[60]1211    x_profile=sum(Asub,1);
1212    y_profile=sum(Asub,2);
1213    [Amax,ind_x_max]=max(x_profile);
1214    [Amax,ind_y_max]=max(y_profile);
[61]1215    %sub-pixel improvement using moments
1216    x_shift=0;
1217    y_shift=0;
[109]1218    %if ind_x_max+2<=2*ind_range_x+1 && ind_x_max-2>=1
1219    if ind_x_max+2<=numel(x_profile) && ind_x_max-2>=1
[61]1220        Atop=x_profile(ind_x_max-2:ind_x_max+2);
1221        x_shift=sum(Atop.*[-2 -1 0 1 2])/sum(Atop);
1222    end
[114]1223    if ind_y_max+2<=numel(y_profile) && ind_y_max-2>=1
[61]1224        Atop=y_profile(ind_y_max-2:ind_y_max+2);
1225        y_shift=sum(Atop.*[-2 -1 0 1 2]')/sum(Atop);
1226    end
[109]1227    Delta(ipoint,1)=(x_shift+ind_x_max+i0min-i0-1)*Dx;%shift from the initial guess
1228    Delta(ipoint,2)=(y_shift+ind_y_max+j0min-j0-1)*Dy;
[60]1229end
1230Tmod=T(:,(1:2))+Delta;
1231[Xpx,Ypx]=px_XYZ(GeometryCalib,Tmod(:,1),Tmod(:,2));
[63]1232for ipoint=1:nbpoints
1233     Coord{ipoint,1}=num2str(T(ipoint,1),4);%display coordiantes with 4 digits
1234     Coord{ipoint,2}=num2str(T(ipoint,2),4);%display coordiantes with 4 digits
[109]1235     Coord{ipoint,3}=num2str(T(ipoint,3),4);%display coordiantes with 4 digits;
1236     Coord{ipoint,4}=num2str(Xpx(ipoint),4);%display coordiantes with 4 digits
1237     Coord{ipoint,5}=num2str(Ypx(ipoint),4);%display coordiantes with 4 digits
[60]1238end
[128]1239Tabchar=cell2tab(Coord(end:-1:1,:),' | ');
[71]1240Tabchar=[Tabchar ;{'......'}];
[60]1241set(handles.ListCoord,'Value',1)
1242set(handles.ListCoord,'String',Tabchar)
[67]1243MenuPlot_Callback(hObject, eventdata, handles)
[60]1244
[71]1245%-----------------------------------------------------------------------
1246function MenuTranslatePoints_Callback(hObject, eventdata, handles)
1247%-----------------------------------------------------------------------
1248%hcalib=get(handles.calib_type,'parent');%handles of the GUI geometry_calib
1249CalibData=get(handles.geometry_calib,'UserData');
1250Tinput=[];%default
1251if isfield(CalibData,'translate')
1252    Tinput=CalibData.translate;
1253end
1254T=translate_points(Tinput);%display translate_points GUI and get shift parameters
1255CalibData.translate=T;
1256set(handles.geometry_calib,'UserData',CalibData)
1257%translation
1258Coord_cell=get(handles.ListCoord,'String');
1259data=read_geometry_calib(Coord_cell);
1260data.Coord(:,1)=T(1)+data.Coord(:,1);
1261data.Coord(:,2)=T(2)+data.Coord(:,2);
1262data.Coord(:,3)=T(3)+data.Coord(:,3);
1263data.Coord(:,[4 5])=data.Coord(:,[4 5]);
1264for i=1:size(data.Coord,1)
1265    for j=1:5
1266          Coord{i,j}=num2str(data.Coord(i,j),4);%phys x,y,z
1267   end
1268end
[128]1269Tabchar=cell2tab(Coord,' | ');
[88]1270Tabchar=[Tabchar; {'.....'}];
[71]1271%set(handles.ListCoord,'Value',1)
1272set(handles.ListCoord,'String',Tabchar)
1273
1274
1275% --------------------------------------------------------------------
1276function MenuRotatePoints_Callback(hObject, eventdata, handles)
1277%hcalib=get(handles.calib_type,'parent');%handles of the GUI geometry_calib
1278CalibData=get(handles.geometry_calib,'UserData');
1279Tinput=[];%default
1280if isfield(CalibData,'rotate')
1281    Tinput=CalibData.rotate;
1282end
1283T=rotate_points(Tinput);%display translate_points GUI and get shift parameters
1284CalibData.rotate=T;
1285set(handles.geometry_calib,'UserData',CalibData)
1286%-----------------------------------------------------
1287%rotation
1288Phi=T(1);
1289O_x=0;%default
1290O_y=0;%default
1291if numel(T)>=2
1292    O_x=T(2);%default
1293end
1294if numel(T)>=3
1295    O_y=T(3);%default
1296end
1297Coord_cell=get(handles.ListCoord,'String');
1298data=read_geometry_calib(Coord_cell);
1299r1=cos(pi*Phi/180);
1300r2=-sin(pi*Phi/180);
1301r3=sin(pi*Phi/180);
1302r4=cos(pi*Phi/180);
1303x=data.Coord(:,1)-O_x;
1304y=data.Coord(:,2)-O_y;
1305data.Coord(:,1)=r1*x+r2*y;
1306data.Coord(:,2)=r3*x+r4*y;
1307% data.Coord(:,[4 5])=data.Coord(:,[4 5]);
1308for i=1:size(data.Coord,1)
1309    for j=1:5
1310          Coord{i,j}=num2str(data.Coord(i,j),4);%phys x,y,z
1311   end
1312end
[128]1313Tabchar=cell2tab(Coord,'| ');
[71]1314Tabchar=[Tabchar;{'......'}];
1315set(handles.ListCoord,'Value',1)
1316set(handles.ListCoord,'String',Tabchar)
1317
[109]1318
1319% %------------------------------------------------------------------------
1320% % --- Executes on button press in rotation.
1321% function rotation_Callback(hObject, eventdata, handles)
1322% %------------------------------------------------------------------------
1323% angle_rot=(pi/180)*str2num(get(handles.Phi,'String'));
1324% Coord_cell=get(handles.ListCoord,'String');
1325% data=read_geometry_calib(Coord_cell);
1326% data.Coord(:,1)=cos(angle_rot)*data.Coord(:,1)+sin(angle_rot)*data.Coord(:,2);
1327% data.Coord(:,1)=-sin(angle_rot)*data.Coord(:,1)+cos(angle_rot)*data.Coord(:,2);
1328% set(handles.XObject,'String',num2str(data.Coord(:,1),4));
1329% set(handles.YObject,'String',num2str(data.Coord(:,2),4));
1330
1331
[108]1332%------------------------------------------------------------------------
1333% image transform from px to phys
1334%INPUT:
1335%Zindex: index of plane
[114]1336% function [A_out,Rangx,Rangy]=phys_Ima(A,Calib,ZIndex)
1337% %------------------------------------------------------------------------
1338% xcorner=[];
1339% ycorner=[];
1340% npx=[];
1341% npy=[];
1342% siz=size(A)
1343% npx=[npx siz(2)];
1344% npy=[npy siz(1)]
1345% xima=[0.5 siz(2)-0.5 0.5 siz(2)-0.5];%image coordinates of corners
1346% yima=[0.5 0.5 siz(1)-0.5 siz(1)-0.5];
1347% [xcorner,ycorner]=phys_XYZ(Calib,xima,yima,ZIndex);%corresponding physical coordinates
1348% Rangx(1)=min(xcorner);
1349% Rangx(2)=max(xcorner);
1350% Rangy(2)=min(ycorner);
1351% Rangy(1)=max(ycorner);
1352% test_multi=(max(npx)~=min(npx)) | (max(npy)~=min(npy));
1353% npx=max(npx);
1354% npy=max(npy);
1355% x=linspace(Rangx(1),Rangx(2),npx);
1356% y=linspace(Rangy(1),Rangy(2),npy);
1357% [X,Y]=meshgrid(x,y);%grid in physical coordiantes
1358% vec_B=[];
1359%
1360% zphys=0; %default
1361% if isfield(Calib,'SliceCoord') %.Z= index of plane
1362%    SliceCoord=Calib.SliceCoord(ZIndex,:);
1363%    zphys=SliceCoord(3); %to generalize for non-parallel planes
1364% end
1365% [XIMA,YIMA]=px_XYZ(Calib,X,Y,zphys);%corresponding image indices for each point in the real space grid
1366% XIMA=reshape(round(XIMA),1,npx*npy);%indices reorganized in 'line'
1367% YIMA=reshape(round(YIMA),1,npx*npy);
1368% flagin=XIMA>=1 & XIMA<=npx & YIMA >=1 & YIMA<=npy;%flagin=1 inside the original image
1369% testuint8=isa(A,'uint8');
1370% testuint16=isa(A,'uint16');
1371% if numel(siz)==2 %(B/W images)
1372%     vec_A=reshape(A,1,npx*npy);%put the original image in line
1373%     ind_in=find(flagin);
1374%     ind_out=find(~flagin);
1375%     ICOMB=((XIMA-1)*npy+(npy+1-YIMA));
1376%     ICOMB=ICOMB(flagin);%index corresponding to XIMA and YIMA in the aligned original image vec_A
1377%     vec_B(ind_in)=vec_A(ICOMB);
1378%     vec_B(ind_out)=zeros(size(ind_out));
1379%     A_out=reshape(vec_B,npy,npx);%new image in real coordinates
1380% elseif numel(siz)==3     
1381%     for icolor=1:siz(3)
1382%         vec_A=reshape(A{icell}(:,:,icolor),1,npx*npy);%put the original image in line
1383%         ind_in=find(flagin);
1384%         ind_out=find(~flagin);
1385%         ICOMB=((XIMA-1)*npy+(npy+1-YIMA));
1386%         ICOMB=ICOMB(flagin);%index corresponding to XIMA and YIMA in the aligned original image vec_A
1387%         vec_B(ind_in)=vec_A(ICOMB);
1388%         vec_B(ind_out)=zeros(size(ind_out));
1389%         A_out(:,:,icolor)=reshape(vec_B,npy,npx);%new image in real coordinates
1390%     end
1391% end
1392% if testuint8
1393%     A_out=uint8(A_out);
1394% end
1395% if testuint16
1396%     A_out=uint16(A_out);
1397% end
[60]1398
[108]1399%------------------------------------------------------------------------
1400% pointwise transform from px to phys
[60]1401%INPUT:
1402%Z: index of plane
[114]1403% function [Xphys,Yphys,Zphys]=phys_XYZ(Calib,X,Y,Z)
1404% %------------------------------------------------------------------------
1405% if exist('Z','var')& isequal(Z,round(Z))& Z>0 & isfield(Calib,'SliceCoord')&length(Calib.SliceCoord)>=Z
1406%     Zindex=Z;
1407%     Zphys=Calib.SliceCoord(Zindex,3);%GENERALISER AUX CAS AVEC ANGLE
1408% else
1409%     Zphys=0;
1410% end
1411% if ~exist('X','var')||~exist('Y','var')
1412%     Xphys=[];
1413%     Yphys=[];%default
1414%     return
1415% end
1416% Xphys=X;%default
1417% Yphys=Y;
1418% %image transform
1419% if isfield(Calib,'R')
1420%     R=(Calib.R)';
1421%     Dx=R(5)*R(7)-R(4)*R(8);
1422%     Dy=R(1)*R(8)-R(2)*R(7);
1423%     D0=Calib.f*(R(2)*R(4)-R(1)*R(5));
1424%     Z11=R(6)*R(8)-R(5)*R(9);
1425%     Z12=R(2)*R(9)-R(3)*R(8); 
1426%     Z21=R(4)*R(9)-R(6)*R(7);
1427%     Z22=R(3)*R(7)-R(1)*R(9);
1428%     Zx0=R(3)*R(5)-R(2)*R(6);
1429%     Zy0=R(1)*R(6)-R(3)*R(4);
1430%     A11=R(8)*Calib.Ty-R(5)*Calib.Tz+Z11*Zphys;
1431%     A12=R(2)*Calib.Tz-R(8)*Calib.Tx+Z12*Zphys;
1432%     A21=-R(7)*Calib.Ty+R(4)*Calib.Tz+Z21*Zphys;
1433%     A22=-R(1)*Calib.Tz+R(7)*Calib.Tx+Z11*Zphys;
1434%     X0=Calib.f*(R(5)*Calib.Tx-R(2)*Calib.Ty+Zx0*Zphys);
1435%     Y0=Calib.f*(-R(4)*Calib.Tx+R(1)*Calib.Ty+Zy0*Zphys);
1436%         %px to camera:
1437%     Xd=(Calib.dpx/Calib.sx)*(X-Calib.Cx); % sensor coordinates
1438%     Yd=Calib.dpy*(Y-Calib.Cy);
1439%     dist_fact=1+Calib.kappa1*(Xd.*Xd+Yd.*Yd); %distortion factor
1440%     Xu=dist_fact.*Xd;%undistorted sensor coordinates
1441%     Yu=dist_fact.*Yd;
1442%     denom=Dx*Xu+Dy*Yu+D0;
1443%     % denom2=denom.*denom;
1444%     Xphys=(A11.*Xu+A12.*Yu+X0)./denom;%world coordinates
1445%     Yphys=(A21.*Xu+A22.*Yu+Y0)./denom;
1446% end
[109]1447
1448
1449% --------------------------------------------------------------------
1450function MenuImportPoints_Callback(hObject, eventdata, handles)
[121]1451fileinput=browse_xml(hObject, eventdata, handles);
[109]1452if isempty(fileinput)
1453    return
1454end
1455[s,errormsg]=imadoc2struct(fileinput,'GeometryCalib');
1456GeometryCalib=s.GeometryCalib;
[114]1457%GeometryCalib=load_calib(hObject, eventdata, handles)
[116]1458calib=reshape(GeometryCalib.PointCoord,[],1);
[109]1459for ilist=1:numel(calib)
1460    CoordCell{ilist}=num2str(calib(ilist));
1461end
1462CoordCell=reshape(CoordCell,[],5);
[128]1463Tabchar=cell2tab(CoordCell,' | ');%transform cells into table ready for display
[109]1464Tabchar=[Tabchar;{'......'}];
1465set(handles.ListCoord,'Value',1)
1466set(handles.ListCoord,'String',Tabchar)
1467MenuPlot_Callback(handles.geometry_calib, [], handles)
1468
1469% -----------------------------------------------------------------------
1470function MenuImportIntrinsic_Callback(hObject, eventdata, handles)
1471%------------------------------------------------------------------------
1472fileinput=browse_xml(hObject, eventdata, handles);
1473if isempty(fileinput)
1474    return
1475end
1476[s,errormsg]=imadoc2struct(fileinput,'GeometryCalib');
1477GeometryCalib=s.GeometryCalib;
[114]1478display_intrinsic(GeometryCalib,handles)
[109]1479
1480% -----------------------------------------------------------------------
1481function MenuImportAll_Callback(hObject, eventdata, handles)
1482%------------------------------------------------------------------------
[121]1483fileinput=browse_xml(hObject, eventdata, handles);
[109]1484if ~isempty(fileinput)
1485    loadfile(handles,fileinput)
1486end
1487
1488% -----------------------------------------------------------------------
1489% --- Executes on menubar option Import/Grid file: introduce previous grid files
1490function MenuGridFile_Callback(hObject, eventdata, handles)
1491% -----------------------------------------------------------------------
[121]1492inputfile=browse_xml(hObject, eventdata, handles);
[109]1493listfile=get(handles.coord_files,'string');
1494if isequal(listfile,{''})
1495    listfile={inputfile};
1496else
[121]1497    listfile=[listfile;{inputfile}];%update the list of coord files
[109]1498end
1499set(handles.coord_files,'string',listfile);
1500
1501%------------------------------------------------------------------------
[114]1502% --- 'key_press_fcn:' function activated when a key is pressed on the keyboard
1503function key_press_fcn(hObject,eventdata,handles)
1504%------------------------------------------------------------------------
1505xx=double(get(handles.geometry_calib,'CurrentCharacter')); %get the keyboard character
1506if ismember(xx,[8 127])%backspace or delete
1507    Coord_cell=get(handles.ListCoord,'String');
1508    val=get(handles.ListCoord,'Value');
1509     if max(val)<numel(Coord_cell) % the last element '...' has not been selected
1510        Coord_cell(val)=[];%remove the selected line
1511        set(handles.ListCoord,'Value',min(val))
1512        set(handles.ListCoord,'String',Coord_cell)         
1513        ListCoord_Callback(hObject, eventdata, handles)
1514        MenuPlot_Callback(hObject,eventdata,handles)
1515     end
1516end
1517
1518%------------------------------------------------------------------------
[109]1519function fileinput=browse_xml(hObject, eventdata, handles)
1520%------------------------------------------------------------------------
1521fileinput=[];%default
1522oldfile=''; %default
[116]1523UserData=get(handles.geometry_calib,'UserData');
[114]1524if isfield(UserData,'XmlInputFile')
1525    oldfile=UserData.XmlInputFile;
[109]1526end
1527[FileName, PathName, filterindex] = uigetfile( ...
1528       {'*.xml;*.mat', ' (*.xml,*.mat)';
1529       '*.xml',  '.xml files '; ...
1530        '*.mat',  '.mat matlab files '}, ...
1531        'Pick a file',oldfile);
1532fileinput=[PathName FileName];%complete file name
1533testblank=findstr(fileinput,' ');%look for blanks
1534if ~isempty(testblank)
1535    msgbox_uvmat('ERROR','forbidden input file name or path: no blank character allowed')
1536    return
1537end
1538sizf=size(fileinput);
1539if (~ischar(fileinput)||~isequal(sizf(1),1)),return;end
[114]1540UserData.XmlInputFile=fileinput;
[109]1541set(handles.geometry_calib,'UserData',UserData)%record current file foer further use of browser
1542
1543% -----------------------------------------------------------------------
[128]1544function Heading=loadfile(handles,fileinput)
[109]1545%------------------------------------------------------------------------
[128]1546Heading=[];%default
[121]1547[s,errormsg]=imadoc2struct(fileinput,'GeometryCalib');
[128]1548if ~isempty(errormsg)
1549    msgbox_uvmat('ERROR',['Error for reading ' fileinput ': '  errormsg])
1550    return
1551end
1552if ~isempty(s.Heading)
1553    Heading=s.Heading;
1554end
1555   
[116]1556GeometryCalib=s.GeometryCalib;
[114]1557fx=1;fy=1;Cx=0;Cy=0;kc=0; %default
1558%     Tabchar={};
1559CoordCell={};
1560%     kc=0;%default
1561%     f1=1000;
1562%     f2=1000;
1563%     hhuvmat=guidata(findobj(allchild(0),'Name','uvmat'));
1564%     Cx=str2num(get(hhuvmat.npx,'String'))/2;
1565%     Cy=str2num(get(hhuvmat.npy,'String'))/2;
1566Tabchar={};%default
1567val_cal=1;%default
1568if ~isempty(GeometryCalib)
1569    % choose the calibration option
1570    if isfield(GeometryCalib,'CalibrationType')
1571       calib_list=get(handles.calib_type,'String');
1572       for ilist=1:numel(calib_list)
1573           if strcmp(calib_list{ilist},GeometryCalib.CalibrationType)
1574               val_cal=ilist;
1575               break
1576           end
1577       end
1578    end
1579    display_intrinsic(GeometryCalib,handles)%intrinsic param
1580    %extrinsic param
1581    if isfield(GeometryCalib,'Tx_Ty_Tz')
1582        Tx_Ty_Tz=GeometryCalib.Tx_Ty_Tz;
1583        set(handles.Tx,'String',num2str(GeometryCalib.Tx_Ty_Tz(1),4))
1584        set(handles.Ty,'String',num2str(GeometryCalib.Tx_Ty_Tz(2),4))
1585        set(handles.Tz,'String',num2str(GeometryCalib.Tx_Ty_Tz(3),4))
1586    end
1587    if isfield(GeometryCalib,'omc')
1588        set(handles.Phi,'String',num2str(GeometryCalib.omc(1),4))
1589        set(handles.Theta,'String',num2str(GeometryCalib.omc(2),4))
1590        set(handles.Psi,'String',num2str(GeometryCalib.omc(3),4))
1591    end
[116]1592    calib=reshape(GeometryCalib.PointCoord,[],1);
[109]1593    for ilist=1:numel(calib)
1594        CoordCell{ilist}=num2str(calib(ilist));
1595    end
1596    CoordCell=reshape(CoordCell,[],5);
[128]1597    Tabchar=cell2tab(CoordCell,' | ');%transform cells into table ready for display
[109]1598    MenuPlot_Callback(handles.geometry_calib, [], handles)
1599end
[114]1600set(handles.calib_type,'Value',val_cal)
[109]1601Tabchar=[Tabchar;{'......'}];
[114]1602set(handles.ListCoord,'Value',1)
1603set(handles.ListCoord,'String',Tabchar)
1604
[109]1605if isempty(CoordCell)% allow mouse action by default in the absence of input points
1606    set(handles.edit_append,'Value',1)
1607    set(handles.edit_append,'BackgroundColor',[1 1 0])
1608else % does not allow mouse action by default in the presence of input points
1609    set(handles.edit_append,'Value',0)
1610    set(handles.edit_append,'BackgroundColor',[0.7 0.7 0.7])
1611end
1612
[114]1613%------------------------------------------------------------------------
1614%---display calibration intrinsic parameters
1615function display_intrinsic(GeometryCalib,handles)
1616%------------------------------------------------------------------------
1617fx=[];
1618fy=[];
1619if isfield(GeometryCalib,'fx_fy')
1620    fx=GeometryCalib.fx_fy(1);
1621    fy=GeometryCalib.fx_fy(2);
1622end
1623Cx_Cy=[0 0];%default
1624if isfield(GeometryCalib,'Cx_Cy')
1625    Cx_Cy=GeometryCalib.Cx_Cy;
1626end
1627kc=0;
1628if isfield(GeometryCalib,'kc')
[121]1629    kc=GeometryCalib.kc; %* GeometryCalib.focal*GeometryCalib.focal;
[114]1630end
1631set(handles.fx,'String',num2str(fx,5))
1632set(handles.fy,'String',num2str(fy,5))
1633set(handles.Cx,'String',num2str(Cx_Cy(1),'%1.1f'))
1634set(handles.Cy,'String',num2str(Cx_Cy(2),'%1.1f'))
1635set(handles.kc,'String',num2str(kc,'%1.4f'))
[109]1636
1637
Note: See TracBrowser for help on using the repository browser.