source: trunk/src/geometry_calib.m @ 1175

Last change on this file since 1175 was 1174, checked in by sommeria, 2 weeks ago

Import export added for geometrycalib

File size: 68.3 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%=======================================================================
13% Copyright 2008-2024, LEGI UMR 5519 / CNRS UGA G-INP, Grenoble, France
14%   http://www.legi.grenoble-inp.fr
15%   Joel.Sommeria - Joel.Sommeria (A) univ-grenoble-alpes.fr
16%
17%     This file is part of the toolbox UVMAT.
18%
19%     UVMAT is free software; you can redistribute it and/or modify
20%     it under the terms of the GNU General Public License as published
21%     by the Free Software Foundation; either version 2 of the license,
22%     or (at your option) any later version.
23%
24%     UVMAT is distributed in the hope that it will be useful,
25%     but WITHOUT ANY WARRANTY; without even the implied warranty of
26%     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
27%     GNU General Public License (see LICENSE.txt) for more details.
28%=======================================================================
29
30function varargout = geometry_calib(varargin)
31% GEOMETRY_CALIB M-file for geometry_calib.fig
32%      GEOMETRY_CALIB, by itself, creates a MenuCoord GEOMETRY_CALIB or raises the existing
33%      singleton*.
34%
35%      H = GEOMETRY_CALIB returns the handle to a MenuCoord GEOMETRY_CALIB or the handle to
36%      the existing singleton*.
37%
38%      GEOMETRY_CALIB('CALLBACK',hObject,eventData,handles,...) calls the local
39%      function named CALLBACK in GEOMETRY_CALIB.M with the given input arguments.
40%
41%      GEOMETRY_CALIB('Property','Value',...) creates a MenuCoord GEOMETRY_CALIB or raises the
42%      existing singleton*.  Starting from the left, property value pairs are
43%      applied to the GUI before geometry_calib_OpeningFunction gets called.  An
44%      unrecognized property name or invalid value makes property application
45%      stop.  All inputs are passed to geometry_calib_OpeningFcn via varargin.
46%
47%      *See GUI Options on GUIDE's Tools menu.  Choose "GUI allows only one
48%      instance to run (singleton)".
49%
50% See also: GUIDE, GUIDATA, GUIHANDLES
51
52% Edit the above text to modify the response to help geometry_calib
53
54% Last Modified by GUIDE v2.5 05-Feb-2022 22:24:55
55
56% Begin initialization code - DO NOT edit
57gui_Singleton = 1;
58gui_State = struct('gui_Name',       mfilename, ...
59    'gui_Singleton',  gui_Singleton, ...
60    'gui_OpeningFcn', @geometry_calib_OpeningFcn, ...
61    'gui_OutputFcn',  @geometry_calib_OutputFcn, ...
62    'gui_LayoutFcn',  [] , ...
63    'gui_Callback',   []);
64if nargin
65    [pp,ff]=fileparts(which(varargin{1})); % name of the input file
66    if strcmp(ff,mfilename)% if we are activating a sub-function of geometry_calib
67        % ~isempty(regexp(varargin{1},'_Callback','once'))
68        gui_State.gui_Callback = str2func(varargin{1});
69    end
70end
71
72if nargout
73    [varargout{1:nargout}] = gui_mainfcn(gui_State, varargin{:});
74else
75    gui_mainfcn(gui_State, varargin{:});
76end
77% End initialization code - DO NOT edit
78
79
80% --- Executes just before geometry_calib is made visible.
81%INPUT:
82%handles: handles of the geometry_calib interface elements
83% PlotHandles: set of handles of the elements contolling the plotting
84% parameters on the uvmat interface (obtained by 'get_plot_handle.m')
85%------------------------------------------------------------------------
86function geometry_calib_OpeningFcn(hObject, eventdata, handles,inputfile)
87%------------------------------------------------------------------------
88
89handles.output = hObject;
90% Update handles structure
91guidata(hObject, handles);
92set(hObject,'DeleteFcn',{@closefcn})%
93
94%% position
95set(0,'Unit','pixels')
96ScreenSize=get(0,'ScreenSize');% get the size of the screen, to put the fig on the upper right
97Left=ScreenSize(3)- 460; %right edge close to the right, with margin=40 (GUI width=420 px)
98if ScreenSize(4)>920
99    Height=840;%default height of the GUI
100    Bottom=ScreenSize(4)-Height-40; %put fig at top right
101else
102    Height=ScreenSize(4)-80;
103    Bottom=40; % GUI lies o the screen bottom (with margin =40)
104end
105set(handles.calib_type,'Position',[1 Height-40 194 30])%  rank 1
106set(handles.APPLY,'Position',[197 Height-40 110 30])%  rank 1
107set(handles.Replicate,'Position',[309 Height-40 110 30])%  rank 1
108set(handles.Intrinsic,'Position',[1 Height-40-2-92 418 92])%  rank 2
109set(handles.Extrinsic,'Position',[1 Height-40-4-92-75 418 75])%  rank 3
110set(handles.PointLists,'Position',[1 Height-40-6-92-75-117 418 117]) %  rank 4
111set(handles.CheckEnableMouse,'Position',[3 Height-362 208 30])%  rank 5
112set(handles.PLOT,'Position',[3 Height-394 120 30])%  rank 6
113set(handles.Copy,'Position',[151 Height-394 120 30])%  rank 6
114set(handles.ClearAll,'Position',[297 Height-394 120 30])%  rank 6
115set(handles.ClearPoint,'Position',[297 Height-362 120 30])%  rank 6
116set(handles.CoordLine,'Position',[211 Height-362 86 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_order4';'3D_extrinsic'})
125if exist('inputfile','var')&& ~isempty(inputfile)
126    [RootPath,SubDir,RootFile,tild,tild,tild,tild,FileExt]=fileparts_uvmat(inputfile);
127    struct.XmlInputFile=find_imadoc(RootPath,SubDir);
128    set(handles.ListCoord,'Data',[])
129    if exist(struct.XmlInputFile,'file')
130        Heading=loadfile(handles,struct.XmlInputFile);% load data from the xml file and fill the GUI
131        if isfield(Heading,'Campaign')&& ischar(Heading.Campaign)
132            struct.Campaign=Heading.Campaign;
133        end
134    end
135    set(hObject,'UserData',struct)
136end
137
138%------------------------------------------------------------------------
139% --- Outputs from this function are returned to the command line.
140function varargout = geometry_calib_OutputFcn(~, eventdata, handles)
141%------------------------------------------------------------------------
142% Get default command line output from handles structure
143varargout{1} = handles.output;
144varargout{2}=handles;
145               
146%------------------------------------------------------------------------
147% executed when closing: set the parent interface button to value 0
148function closefcn(gcbo,eventdata)
149%------------------------------------------------------------------------
150huvmat=findobj(allchild(0),'Name','uvmat');
151if ~isempty(huvmat)
152    handles=guidata(huvmat);
153    set(handles.MenuCalib,'Checked','off')
154    hobject=findobj(handles.PlotAxes,'tag','calib_points');
155    if ~isempty(hobject)
156        delete(hobject)
157    end
158    hobject=findobj(handles.PlotAxes,'tag','calib_marker');
159    if ~isempty(hobject)
160        delete(hobject)
161    end
162end
163               
164%------------------------------------------------------------------------
165% --- Executes on button press APPLY (used to launch the calibration).
166function APPLY_Callback(hObject, eventdata, handles)
167set(handles.CheckEnableMouse,'Value',0)% desactivate mouse (to avoid spurious creation of new points)
168
169%------------------------------------------------------------------------
170%% look for the GUI uvmat and check for an image as input
171set(handles.APPLY,'BackgroundColor',[1 1 0])% paint APPLY button in yellow to show activation
172huvmat=findobj(allchild(0),'Name','uvmat');% look for the GUI uvmat
173hhuvmat=guidata(huvmat);%handles of elements in the GUI uvmat
174if ~strcmp(get(hhuvmat.Scalar,'Visible'),'on')
175    msgbox_uvmat('ERROR','An image needs to be opened in uvmat for calibration')
176    return
177end
178
179RootPath='';
180if ~isempty(hhuvmat.RootPath)&& ~isempty(hhuvmat.RootFile)
181    RootPath=get(hhuvmat.RootPath,'String');% path to the currently displayed image
182    %SubDirBase=regexprep(get(hhuvmat.SubDir,'String'),'\..+$','');
183    outputfile=[fullfile(RootPath,get(hhuvmat.SubDir,'String')) '.xml'];%xml file associated with the currently displayed image
184else
185    question={'save the calibration data and point coordinates in'};
186    def={fullfile(RootPath,'ObjectCalib.xml')};
187    options.Resize='on';
188    answer=inputdlg(question,'',1,def,options);
189    outputfile=answer{1};
190end
191
192%% read coordinates of the calibration points: Coord(:,1-3) in phys, Coord(:,4-5) image
193Coord=get(handles.ListCoord,'Data');
194if isempty(Coord)
195    msgbox_uvmat('ERROR','introduce calibration points in the table Coord, or use the Tools Set scale')
196    return
197end
198
199
200%% read the type of calibration
201calib_cell=get(handles.calib_type,'String');
202val=get(handles.calib_type,'Value');
203CalibFcn=['calib_' calib_cell{val}];
204
205%% read the intrinsic parameters
206Intrinsic.Npx=str2num(get(hhuvmat.num_Npx,'String'));
207Intrinsic.Npy=str2num(get(hhuvmat.num_Npy,'String'));
208Intrinsic.coord_files=get(handles.ListCoordFiles,'String');
209Intrinsic.fx=str2num(get(handles.fx,'String'));
210Intrinsic.fy=str2num(get(handles.fy,'String'));
211Intrinsic.kc=str2num(get(handles.kc,'String'));
212Intrinsic.Cx=str2num(get(handles.Cx,'String'));
213Intrinsic.Cy=str2num(get(handles.Cy,'String'));
214if isempty(Intrinsic.kc)
215    Intrinsic.kc=0;
216end
217if isempty(Intrinsic.Cx)||isempty(Intrinsic.Cy)
218    Intrinsic.Cx=Intrinsic.Npx/2;
219    Intrinsic.Cy=Intrinsic.Npy/2;
220end
221
222%% apply to cropped images if requested
223if get(handles.Replicate,'Value')
224    answer=msgbox_uvmat('INPUT_Y-N','apply to full images (not cropped)?');
225    if strcmp(answer,'No')
226        prompt = {'npy_lower'};
227        dlg_title = 'remove image the npy_lower image lines (removal of the upper line does not change calibration)';
228        num_lines= 1;
229        def     = {'0'};
230        answer = inputdlg(prompt,dlg_title,num_lines,def);
231        npy_crop=str2num(answer{1});
232        Intrinsic.Npy=Intrinsic.Npy-npy_crop; %size of the filtering window
233        Coord(:,5)=Coord(:,5)-npy_crop;% shift the image ordinates of the calibration points by removing the lower band
234    end
235end
236
237%% Apply calibration
238est_dist=[0;0;0;0;0]; %default
239switch CalibFcn
240    case 'calib_3D_linear'
241        CalibFcn='calib_3D';
242    case 'calib_3D_quadr'
243        est_dist=[1;0;0;0;0];
244        CalibFcn='calib_3D'
245    case 'calib_3D_order4'
246        est_dist=[1;1;0;0;0];
247        CalibFcn='calib_3D'
248end
249
250[GeometryCalib,index,ind_removed]=calibrate(Coord,CalibFcn,est_dist,Intrinsic);% apply calibration
251if isempty(GeometryCalib)
252    return
253end
254
255%% record the coordinate unit
256unitlist=get(handles.CoordUnit,'String');
257unit=unitlist{get(handles.CoordUnit,'value')};
258GeometryCalib.CoordUnit=unit;
259
260%% record the coordinates of the calibration points
261GeometryCalib.SourceCalib.PointCoord=Coord;
262
263%% set the default plane and display the calibration data errors for validation
264answer=msgbox_uvmat('INPUT_Y-N',{'store calibration data';...
265    ['Error rms (along x,y)=' num2str(GeometryCalib.ErrorRms) ' pixels'];...
266    ['Error max (along x,y)=' num2str(GeometryCalib.ErrorMax) ' pixels'];
267    [num2str(numel(ind_removed)) ' points removed with ErrorMax > 3 rms']});
268checkslice=0;
269if strcmp(answer,'Yes') %store the calibration data
270    display_intrinsic(GeometryCalib,handles)%display calibration intrinsic parameters on the GUI geometry_calib
271    display_extrinsic(GeometryCalib,handles)%display calibration extrinsic parameters on the GUI geometry_calib
272    Z=Coord(:,3);
273    if strcmp(calib_cell{val}(1:2),'3D') && isequal(max(Z),min(Z))%set the plane position for 3D (projection) calibration
274        %set the Z position of the reference plane used for calibration
275        answer=msgbox_uvmat('INPUT_Y-N',{['Do you assume that the illuminated plane is at z=' num2str(Z(1)) '?' ] ; 'can be later modified by MenuSetSlice in the upper bar menu of uvmat'});
276        if strcmp(answer,'Yes')
277            Slice.NbSlice=1;
278            Slice.SliceCoord=[0 0 Z(1)];
279            checkslice=1;% will document the item <Slice> in ImaDoc
280        end
281    end
282else
283    GeometryCalib=[];
284    index=1;
285end
286
287if ~isempty(GeometryCalib) % if calibration is not cancelled
288    %%%%% use of the option 'replicate'
289    if get(handles.Replicate,'Value')% if the option replicate is activated
290        %nbcalib=0;
291        %% open the GUI browse_data
292        hbrowse=findobj(allchild(0),'Tag','browse_data');
293        if ~isempty(hbrowse)% look for the GUI browse_data
294            BrowseData=guidata(hbrowse);
295            SourceDir=get(BrowseData.SourceDir,'String');
296            ListExp=get(BrowseData.ListExperiments,'String');
297            ListExp=ListExp(get(BrowseData.ListExperiments,'Value'));% list of selected experiments
298            ListDevices=get(BrowseData.ListDevices,'String');
299            ListDevices=ListDevices(get(BrowseData.ListDevices,'Value'));% list of selected devices
300            ListDataSeries=get(BrowseData.DataSeries,'String');
301            ListDataSeries=ListDataSeries(get(BrowseData.DataSeries,'Value'));% list of selected data series
302            if find(~cellfun('isempty',strfind(ListDataSeries,'.xml')))
303                msgbox_uvmat('ERROR','select folders in browse_data, not xml files');
304                return
305            end
306            if find(~cellfun('isempty',strfind(ListDataSeries,'.')))
307                msgbox_uvmat('WARNING','select folders at the root, without dot (.) in the name');
308            end
309            NbExp=0; % counter of the number of experiments set by the GUI browse_data
310            for iexp=1:numel(ListExp)
311                if ~isempty(regexp(ListExp{iexp},'^\+/'))% if it is a folder
312                    for idevice=1:numel(ListDevices)
313                        if ~isempty(regexp(ListDevices{idevice},'^\+/'))% if it is a folder
314                            for isubdir=1:numel(ListDataSeries)
315                                if ~isempty(regexp(ListDataSeries{isubdir},'^\+/'))% if it is a folder
316                                    lpath= fullfile(SourceDir,regexprep(ListExp{iexp},'^\+/',''),...
317                                        regexprep(ListDevices{idevice},'^\+/',''));
318                                    ldir= regexprep(ListDataSeries{isubdir},'^\+/','');
319                                    if exist(fullfile(lpath,ldir),'dir')
320                                        NbExp=NbExp+1;
321                                        ListPath{NbExp}=lpath;
322                                        ListSubdir{NbExp}=ldir;
323                                        ExpIndex{NbExp}=iexp;
324                                    end
325                                end
326                            end
327                        end
328                    end
329                end
330            end
331            NbErrors=0;
332            for iexp=1:NbExp
333                [check_update,xmlfile,errormsg]=update_imadoc(ListPath{iexp},ListSubdir{iexp},'GeometryCalib',GeometryCalib);% introduce the calibration data in the xml file
334                dispmessage='';
335                if checkslice
336                   [~,~,errormsg]=update_imadoc(ListPath{iexp},ListSubdir{iexp},'Slice',Slice,0);% introduce the slice position in the xml file
337                    dispmessage=' and slice position';
338                end
339                if ~strcmp(errormsg,'')
340                    msgbox_uvmat('ERROR',errormsg);
341                    NbErrors=NbErrors+1;
342                else
343                    if check_update
344                        disp([xmlfile ' updated with calibration parameters' dispmessage]);
345                    else
346                        disp([xmlfile ' created with calibration parameters: no timing defined' dispmessage]);
347                    end
348                end
349            end
350        end
351        msgout=['calibration replicated for ' num2str(NbExp-NbErrors) ' experiments'];
352        if NbErrors~=0
353            msgout={msgout;['error for ' num2str(NbErrors) ' experiments']};
354        end
355        msgbox_uvmat('CONFIMATION',msgout);
356    else
357        %% update the calibration parameters in the currently opened uvmat GUI
358        % if ~exist(outputfile,'file') && ~isempty(SubDirBase) %copy the xml file from the old location if appropriate
359        %     oldxml=[fullfile(RootPath,SubDirBase,get(hhuvmat.RootFile,'String')) '.xml'];
360        %     if exist(oldxml,'file')
361        %         [success,message]=copyfile(oldxml,outputfile);%copy the old xml file to a new one with the new convention
362        %     end
363        % endSlice,
364        [~,~,errormsg]=update_imadoc(RootPath,get(hhuvmat.SubDir,'String'),'GeometryCalib',GeometryCalib);% introduce the calibration data in the xml file
365        if checkslice
366          [~,~,errormsg]=update_imadoc(RootPath,get(hhuvmat.SubDir,'String'),'Slice',Slice,0);% introduce the slice position in the xml file
367        end
368        if ~strcmp(errormsg,'')
369            msgbox_uvmat('ERROR',errormsg);
370        end
371
372        %% display image with new calibration in the currently opened uvmat GUI
373        FieldList=get(hhuvmat.FieldName,'String');
374        val=get(hhuvmat.FieldName,'Value');
375        if strcmp(FieldList{val},'image')
376            set(hhuvmat.CheckFixLimits,'Value',0)% put FixedLimits option to 'off' to plot the whole image
377            UserData=get(handles.geometry_calib,'UserData');
378            UserData.XmlInputFile=outputfile;%save the current xml file name
379            set(handles.geometry_calib,'UserData',UserData)
380            uvmat('InputFileREFRESH_Callback',hObject,eventdata,hhuvmat); %file input with xml reading  in uvmat, show the image in phys coordinates
381            PLOT_Callback(hObject, eventdata, handles)
382            set(handles.CoordLine,'string',num2str(index))
383            Coord=get(handles.ListCoord,'Data');
384            update_calib_marker(Coord(index,:)); %indicate the point with max deviations from phys coord to calibration
385            figure(handles.geometry_calib)% put the GUI geometry_calib in front
386        else
387            msgbox_uvmat('WARNING','open the image to see the effect of the new calibration')
388        end
389    end
390end
391
392set(handles.APPLY,'BackgroundColor',[1 0 0]) % set APPLY button to red color
393
394%------------------------------------------------------------------------
395% --- Executes on button press in Replicate
396function Replicate_Callback(hObject, eventdata, handles)
397% %------------------------------------------------------------------------
398set(handles.CheckEnableMouse,'Value',0)% desactivate mouse (to avoid spurious creation of new points)
399
400if get(handles.Replicate,'Value') %open the GUI browse_data
401    % look for the GUI uvmat and check for an image as input
402    huvmat=findobj(allchild(0),'Name','uvmat');
403    hhuvmat=guidata(huvmat);%handles of elements in the GUI uvmat
404    RootPath=get(hhuvmat.RootPath,'String');
405    SubDir=get(hhuvmat.SubDir,'String');
406    browse_data(fullfile(RootPath,SubDir))
407else
408    hbrowse=findobj(allchild(0),'Tag','browse_data');
409    if ~isempty(hbrowse)
410        delete(hbrowse)
411    end
412end
413%------------------------------------------------------------------------
414% --- activate calibration and store parameters in ouputfile .
415function [GeometryCalib,ind_max,ind_removed]=calibrate(Coord,CalibFcn,est_dist,Intrinsic)
416%------------------------------------------------------------------------
417
418index=[];
419GeometryCalib=[];
420ind_max=[];ind_removed=[];
421% apply the calibration, whose type is selected in  handles.calib_type
422CoordFlip=Coord;
423CoordFlip(:,3)=Coord(:,3);% the calibration function assume a z ccordinate along the camera view, opposite to ours
424if ~isempty(Coord)
425    GeometryCalib=feval(CalibFcn,CoordFlip,est_dist,Intrinsic);
426else
427    msgbox_uvmat('ERROR','No calibration points, abort')
428end
429if isempty(GeometryCalib)
430    return
431end
432
433% estimate calibration error rms and max
434x_ima=Coord(:,4);
435y_ima=Coord(:,5);
436[Xpoints,Ypoints]=px_XYZ(GeometryCalib,[],Coord(:,1),Coord(:,2),Coord(:,3));
437GeometryCalib.ErrorRms(1)=sqrt(mean((Xpoints-x_ima).*(Xpoints-x_ima)));
438GeometryCalib.ErrorRms(2)=sqrt(mean((Ypoints-y_ima).*(Ypoints-y_ima)));
439[ErrorMax(1),index(1)]=max(abs(Xpoints-x_ima));
440[ErrorMax(2),index(2)]=max(abs(Ypoints-y_ima));
441[tild,ind_dim]=max(ErrorMax);
442ind_max=index(ind_dim); % mark the index with maximum deviation
443
444% detect bad calibration points, marked by indices ind_removed, if the
445% difference of actual image coordinates and those given by calibration is
446% greater than 2 pixels and greater than 3 rms.
447check_x=abs(Xpoints-x_ima)>max(2,3*GeometryCalib.ErrorRms(1));
448check_y=abs(Ypoints-y_ima)>max(2,3*GeometryCalib.ErrorRms(2));
449ind_removed=find(check_x | check_y);
450% repeat calibration without the excluded points:
451if ~isempty(ind_removed)
452    Coord(ind_removed,:)=[];
453    CoordFlip=Coord;
454    CoordFlip(:,3)=Coord(:,3);% the calibration function assume a z ccordinate along the camera view, opposite to ours
455    GeometryCalib=feval(CalibFcn,CoordFlip,est_dist,Intrinsic);
456    X=Coord(:,1);
457    Y=Coord(:,2);
458    Z=Coord(:,3);
459    x_ima=Coord(:,4);
460    y_ima=Coord(:,5);
461    [Xpoints,Ypoints]=px_XYZ(GeometryCalib,[],X,Y,Z);
462    GeometryCalib.ErrorRms(1)=sqrt(mean((Xpoints-x_ima).*(Xpoints-x_ima)));
463    GeometryCalib.ErrorRms(2)=sqrt(mean((Ypoints-y_ima).*(Ypoints-y_ima)));
464end
465GeometryCalib.ErrorMax=ErrorMax;
466
467
468%------------------------------------------------------------------------
469% --- determine the parameters for a calibration by an affine function (rescaling and offset, no rotation)
470function GeometryCalib=calib_rescale(Coord,est_dist,Intrinsic)
471%------------------------------------------------------------------------
472X=Coord(:,1);
473Y=Coord(:,2);% Z not used
474x_ima=Coord(:,4);
475y_ima=Coord(:,5);
476[px]=polyfit(X,x_ima,1);
477[py]=polyfit(Y,y_ima,1);
478GeometryCalib.CalibrationType='rescale';
479GeometryCalib.fx_fy=[px(1) py(1)];%.fx_fy corresponds to pxcm along x and y
480GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
481GeometryCalib.Tx_Ty_Tz=[px(2)/px(1) py(2)/py(1) 1];
482GeometryCalib.omc=[0 0 0];
483
484%------------------------------------------------------------------------
485% --- determine the parameters for a calibration by a linear transform matrix (rescale and rotation)
486function GeometryCalib=calib_linear(Coord,est_dist,Intrinsic)
487%------------------------------------------------------------------------
488X=Coord(:,1);
489Y=Coord(:,2);% Z not used
490x_ima=Coord(:,4);
491y_ima=Coord(:,5);
492XY_mat=[ones(size(X)) X Y];
493a_X1=XY_mat\x_ima; %transformation matrix for X
494a_Y1=XY_mat\y_ima;%transformation matrix for X
495R=[a_X1(2),a_X1(3);a_Y1(2),a_Y1(3)];
496epsilon=sign(det(R));
497norm=abs(det(R));
498GeometryCalib.CalibrationType='linear';
499if (a_X1(2)/a_Y1(3))>0
500    GeometryCalib.fx_fy(1)=sqrt((a_X1(2)/a_Y1(3))*norm);
501else
502    GeometryCalib.fx_fy(1)=-sqrt(-(a_X1(2)/a_Y1(3))*norm);
503end
504GeometryCalib.fx_fy(2)=(a_Y1(3)/a_X1(2))*GeometryCalib.fx_fy(1);
505GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
506GeometryCalib.Tx_Ty_Tz=[a_X1(1)/GeometryCalib.fx_fy(1) a_Y1(1)/GeometryCalib.fx_fy(2) 1];
507R(1,:)=R(1,:)/GeometryCalib.fx_fy(1);
508R(2,:)=R(2,:)/GeometryCalib.fx_fy(2);
509R=[R;[0 0]];
510GeometryCalib.R=[R [0;0;-epsilon]];
511GeometryCalib.omc=(180/pi)*[acos(GeometryCalib.R(1,1)) 0 0];
512
513%------------------------------------------------------------------------
514% --- determine the tsai parameters for a view normal to the grid plane
515% NOT USED
516function GeometryCalib=calib_normal(Coord,est_dist,Intrinsic)
517%------------------------------------------------------------------------
518Calib.fx=str2num(get(handles.fx,'String'));
519Calib.fy=str2num(get(handles.fy,'String'));
520Calib.kc=str2num(get(handles.kc,'String'));
521Calib.Cx=str2num(get(handles.Cx,'String'));
522Calib.Cy=str2num(get(handles.Cy,'String'));
523%default
524if isempty(Calib.fx)
525    Calib.fx=25/0.012;
526end
527if isempty(Calib.fy)
528    Calib.fy=25/0.012;
529end
530if isempty(Calib.kc)
531    Calib.kc=0;
532end
533if isempty(Calib.Cx)||isempty(Calib.Cy)
534    huvmat=findobj(allchild(0),'Tag','uvmat');
535    hhuvmat=guidata(huvmat);
536    Calib.Cx=str2num(get(hhuvmat.num_Npx,'String'))/2;
537    Calib.Cx=str2num(get(hhuvmat.num_Npy,'String'))/2;
538end
539%tsai parameters
540Calib.dpx=0.012;%arbitrary
541Calib.dpy=0.012;
542Calib.sx=Calib.fx*Calib.dpx/(Calib.fy*Calib.dpy);
543Calib.f=Calib.fy*Calib.dpy;
544Calib.kappa1=Calib.kc/(Calib.f*Calib.f);
545
546%initial guess
547X=Coord(:,1);
548Y=Coord(:,2);
549Zmean=mean(Coord(:,3));
550x_ima=Coord(:,4)-Calib.Cx;
551y_ima=Coord(:,5)-Calib.Cy;
552XY_mat=[ones(size(X)) X Y];
553a_X1=XY_mat\x_ima; %transformation matrix for X
554a_Y1=XY_mat\y_ima;%transformation matrix for Y
555R=[a_X1(2),a_X1(3),0;a_Y1(2),a_Y1(3),0;0,0,-1];% rotation+ z axis reversal (upward)
556norm=sqrt(det(-R));
557calib_param(1)=0;% quadratic distortion
558calib_param(2)=a_X1(1);
559calib_param(3)=a_Y1(1);
560calib_param(4)=Calib.f/(norm*Calib.dpx)-R(3,3)*Zmean;
561calib_param(5)=angle(a_X1(2)+1i*a_X1(3));
562display(['initial guess=' num2str(calib_param)])
563
564%optimise the parameters: minimisation of error
565calib_param = fminsearch(@(calib_param) error_calib(calib_param,Calib,Coord),calib_param);
566
567GeometryCalib.CalibrationType='tsai_normal';
568GeometryCalib.focal=Calib.f;
569GeometryCalib.dpx_dpy=[Calib.dpx Calib.dpy];
570GeometryCalib.Cx_Cy=[Calib.Cx Calib.Cy];
571GeometryCalib.sx=Calib.sx;
572GeometryCalib.kappa1=calib_param(1);
573GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
574GeometryCalib.Tx_Ty_Tz=[calib_param(2) calib_param(3) calib_param(4)];
575alpha=calib_param(5);
576GeometryCalib.R=[cos(alpha) sin(alpha) 0;-sin(alpha) cos(alpha) 0;0 0 -1];
577%
578% %------------------------------------------------------------------------
579% function GeometryCalib=calib_3D_linear(Coord,Intrinsic)
580% %------------------------------------------------------------------------
581% coord_files=Intrinsic.coord_files;
582% if ischar(Intrinsic.coord_files)
583%     coord_files={coord_files};
584% end
585% if isempty(coord_files{1}) || isequal(coord_files,{''})
586%     coord_files={};
587% end
588% %retrieve the calibration points stored in the files listed in the popup list ListCoordFiles
589% x_1=Coord(:,4:5)';%px coordinates of the ref points
590%
591% nx=Intrinsic.Npx;
592% ny=Intrinsic.Npy;
593% x_1(2,:)=ny-x_1(2,:);%reverse the y image coordinates
594% X_1=Coord(:,1:3)';%phys coordinates of the ref points
595% n_ima=numel(coord_files)+1;
596% if ~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;
605%                     Coord_file=zeros(length(PointCoord),5);%default
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)'';']);
612%                     eval(['x_' num2str(ifile+1) '(2,:)=ny-x_' num2str(ifile+1) '(2,:);' ]);
613%                     eval(['X_' num2str(ifile+1) '=Coord_file(:,1:3)'';']);
614%                 end
615%             end
616%         end
617%     end
618% end
619% n_ima=numel(coord_files)+1;
620% est_dist=[0;0;0;0;0];
621% est_aspect_ratio=0;
622% est_fc=[1;1];
623% center_optim=0;
624% path_uvmat=which('uvmat');% check the path detected for source file uvmat
625% path_UVMAT=fileparts(path_uvmat); %path to UVMAT
626% run(fullfile(path_UVMAT,'toolbox_calib','go_calib_optim'));% apply fct 'toolbox_calib/go_calib_optim'
627% if exist('Rc_1','var')
628%     GeometryCalib.CalibrationType='3D_extrinsic';
629%     GeometryCalib.fx_fy=fc';
630%     GeometryCalib.Cx_Cy=cc';
631%     GeometryCalib.kc=kc(1);
632%     GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
633%     GeometryCalib.Tx_Ty_Tz=Tc_1';
634%     GeometryCalib.R=Rc_1;
635%     GeometryCalib.R(2,1:3)=-GeometryCalib.R(2,1:3);%inversion of the y image coordinate
636%     GeometryCalib.Tx_Ty_Tz(2)=-GeometryCalib.Tx_Ty_Tz(2);%inversion of the y image coordinate
637%     GeometryCalib.Cx_Cy(2)=ny-GeometryCalib.Cx_Cy(2);%inversion of the y image coordinate
638%     GeometryCalib.omc=(180/pi)*omc_1;%angles in degrees
639%     GeometryCalib.ErrorRMS=[];
640%     GeometryCalib.ErrorMax=[];
641% else
642%     msgbox_uvmat('ERROR',['calibration function ' fullfile('toolbox_calib','go_calib_optim') ' did not converge: use multiple views or option 3D_extrinsic'])
643%     GeometryCalib=[];
644% end
645
646%------------------------------------------------------------------------
647function GeometryCalib=calib_3D(Coord,est_dist,Intrinsic)
648%------------------------------------------------------------------
649
650path_uvmat=which('uvmat');% check the path detected for source file uvmat
651path_UVMAT=fileparts(path_uvmat); %path to UVMAT
652huvmat=findobj(allchild(0),'Tag','uvmat');
653hhuvmat=guidata(huvmat);
654if ~strcmp(get(hhuvmat.Scalar,'Visible'),'on')
655    msgbox_uvmat('ERROR','An image needs to be opened in uvmat for calibration')
656    return
657end
658% check_cond=0;
659
660coord_files=Intrinsic.coord_files;
661
662if ischar(coord_files)
663    coord_files={coord_files};
664end
665if isempty(coord_files{1}) || isequal(coord_files,{''})
666    coord_files={};
667end
668
669%retrieve the calibration points stored in the files listed in the popup list ListCoordFiles
670x_1=Coord(:,4:5)';%px coordinates of the ref points
671nx=str2num(get(hhuvmat.num_Npx,'String'));
672ny=str2num(get(hhuvmat.num_Npy,'String'));
673x_1(2,:)=ny-x_1(2,:);%reverse the y image coordinates
674X_1=Coord(:,1:3)';%phys coordinates of the ref points
675n_ima=numel(coord_files)+1;
676if ~isempty(coord_files)
677    msgbox_uvmat('CONFIRMATION',['The xy coordinates of the calibration points in ' num2str(n_ima) ' planes will be used'])
678    for ifile=1:numel(coord_files)
679        t=xmltree(coord_files{ifile});
680        s=convert(t);%convert to matlab structure
681        if isfield(s,'GeometryCalib')
682            if isfield(s.GeometryCalib,'SourceCalib')
683                if isfield(s.GeometryCalib.SourceCalib,'PointCoord')
684                    PointCoord=s.GeometryCalib.SourceCalib.PointCoord;
685                    Coord_file=zeros(length(PointCoord),5);%default
686                    for i=1:length(PointCoord)
687                        line=str2num(PointCoord{i});
688                        Coord_file(i,4:5)=line(4:5);%px x
689                        Coord_file(i,1:3)=line(1:3);%phys x
690                    end
691                    eval(['x_' num2str(ifile+1) '=Coord_file(:,4:5)'';']);
692                    eval(['x_' num2str(ifile+1) '(2,:)=ny-x_' num2str(ifile+1) '(2,:);' ]);
693                    eval(['X_' num2str(ifile+1) '=Coord_file(:,1:3)'';']);
694                end
695            end
696        end
697    end
698end
699n_ima=numel(coord_files)+1;
700%est_dist=[1;0;0;0;0];
701est_aspect_ratio=1;
702center_optim=0;
703run(fullfile(path_UVMAT,'toolbox_calib','go_calib_optim'));% apply fct 'toolbox_calib/go_calib_optim'
704
705if exist('Rc_1','var')
706    GeometryCalib.CalibrationType='3D_extrinsic';
707    GeometryCalib.fx_fy=fc';
708    GeometryCalib.Cx_Cy=cc';
709    ind_kc=find(kc);
710    if ~isempty(ind_kc)
711    GeometryCalib.kc=(kc(ind_kc))';%include cases quadr (one value kc(1)) and order_4 (2 values for kc)
712    end
713    GeometryCalib.CoordUnit=[];% default value, to be updated by the calling function
714    GeometryCalib.Tx_Ty_Tz=Tc_1';
715    GeometryCalib.R=Rc_1;
716    GeometryCalib.R(2,1:3)=-GeometryCalib.R(2,1:3);%inversion of the y image coordinate
717    GeometryCalib.Tx_Ty_Tz(2)=-GeometryCalib.Tx_Ty_Tz(2);%inversion of the y image coordinate
718    GeometryCalib.Cx_Cy(2)=ny-GeometryCalib.Cx_Cy(2);%inversion of the y image coordinate
719    GeometryCalib.omc=(180/pi)*omc_1;%angles in degrees
720    GeometryCalib.ErrorRMS=[];
721    GeometryCalib.ErrorMax=[];
722else
723    msgbox_uvmat('ERROR',['calibration function ' fullfile('toolbox_calib','go_calib_optim') ' did not converge: use multiple views or option 3D_extrinsic'])
724    GeometryCalib=[];
725end
726
727%------------------------------------------------------------------------
728function [GeometryCalib,errormsg]=calib_3D_extrinsic(Coord,est_dist, Intrinsic)
729%------------------------------------------------------------------
730errormsg='';
731path_uvmat=which('geometry_calib');% check the path detected for source file uvmat
732path_UVMAT=fileparts(path_uvmat); %path to UVMAT
733x_1=double(Coord(:,4:5)');%image coordinates
734X_1=double(Coord(:,1:3)');% phys coordinates
735huvmat=findobj(allchild(0),'Tag','uvmat');
736hhuvmat=guidata(huvmat);
737if ~strcmp(get(hhuvmat.Scalar,'Visible'),'on')
738    msgbox_uvmat('ERROR','An image needs to be opened in uvmat for calibration')
739    return
740end
741ny=str2double(get(hhuvmat.num_Npy,'String'));
742x_1(2,:)=ny-x_1(2,:);%reverse the y image coordinates
743%n_ima=1;
744GeometryCalib.CalibrationType='3D_extrinsic';
745fx=Intrinsic.fx;
746fy=Intrinsic.fy;
747Cx=Intrinsic.Cx;
748Cy=Intrinsic.Cy;
749kc=Intrinsic.kc;
750if isempty(fx)
751    errormsg='focal length fx needs to be introduced';
752elseif isempty(fy)
753    errormsg='focal length fy needs to be introduced';
754elseif isempty(Cx)
755    errormsg='shift Cx to image centre needs to be introduced';
756elseif isempty(Cy)
757    errormsg='shift Cy to image centre needs to be introduced';
758end
759if ~isempty(errormsg)
760    GeometryCalib=[];
761     msgbox_uvmat('ERROR',errormsg)
762     return
763end
764GeometryCalib.fx_fy(1)=fx;
765GeometryCalib.fx_fy(2)=fy;
766GeometryCalib.Cx_Cy(1)=Cx;
767GeometryCalib.Cx_Cy(2)=Cy;
768GeometryCalib.kc=kc;
769fct_path=fullfile(path_UVMAT,'toolbox_calib');
770addpath(fct_path)
771GeometryCalib.Cx_Cy(2)=ny-GeometryCalib.Cx_Cy(2);%reverse Cx_Cy(2) for calibration (inversion of px ordinate)
772kc=zeros(1,5);
773kc(1:numel(GeometryCalib.kc))=GeometryCalib.kc;
774[omc,Tc1,Rc1,H,x,ex,JJ] = compute_extrinsic(x_1,X_1,...
775    (GeometryCalib.fx_fy)',GeometryCalib.Cx_Cy',kc);
776rmpath(fct_path);
777GeometryCalib.CoordUnit='';% default value, to be updated by the calling function
778GeometryCalib.Tx_Ty_Tz=Tc1';
779%inversion of z axis
780GeometryCalib.R=Rc1;
781GeometryCalib.R(2,1:3)=-GeometryCalib.R(2,1:3);%inversion of the y image coordinate
782GeometryCalib.Tx_Ty_Tz(2)=-GeometryCalib.Tx_Ty_Tz(2);%inversion of the y image coordinate
783GeometryCalib.Cx_Cy(2)=ny-GeometryCalib.Cx_Cy(2);%inversion of the y image coordinate
784GeometryCalib.omc=(180/pi)*omc';
785
786%------------------------------------------------------------------------
787%function GeometryCalib=calib_tsai_heikkila(Coord)
788% TEST: NOT IMPLEMENTED
789%------------------------------------------------------------------
790% path_uvmat=which('uvmat');% check the path detected for source file uvmat
791% path_UVMAT=fileparts(path_uvmat); %path to UVMAT
792% path_calib=fullfile(path_UVMAT,'toolbox_calib_heikkila');
793% addpath(path_calib)
794% npoints=size(Coord,1);
795% Coord(:,1:3)=10*Coord(:,1:3);
796% Coord=[Coord zeros(npoints,2) -ones(npoints,1)];
797% [par,pos,iter,res,er,C]=cacal('dalsa',Coord);
798% GeometryCalib.CalibrationType='tsai';
799% GeometryCalib.focal=par(2);
800
801
802%------------------------------------------------------------------------
803% --- determine the rms of calibration error
804function ErrorRms=error_calib(calib_param,Calib,Coord)
805%------------------------------------------------------------------------
806%calib_param: vector of free calibration parameters (to optimise)
807%Calib: structure of the given calibration parameters
808%Coord: list of phys coordinates (columns 1-3, and pixel coordinates (columns 4-5)
809Calib.f=25;
810Calib.dpx=0.012;
811Calib.dpy=0.012;
812Calib.sx=1;
813Calib.Cx=512;
814Calib.Cy=512;
815Calib.kappa1=calib_param(1);
816Calib.Tx=calib_param(2);
817Calib.Ty=calib_param(3);
818Calib.Tz=calib_param(4);
819alpha=calib_param(5);
820Calib.R=[cos(alpha) sin(alpha) 0;-sin(alpha) cos(alpha) 0;0 0 -1];
821
822X=Coord(:,1);
823Y=Coord(:,2);
824Z=Coord(:,3);
825x_ima=Coord(:,4);
826y_ima=Coord(:,5);
827[Xpoints,Ypoints]=px_XYZ(Calib,[],X,Y,Z);
828ErrorRms(1)=sqrt(mean((Xpoints-x_ima).*(Xpoints-x_ima)));
829ErrorRms(2)=sqrt(mean((Ypoints-y_ima).*(Ypoints-y_ima)));
830ErrorRms=mean(ErrorRms);
831
832
833%------------------------------------------------------------------------
834% --- Executes on button press in STORE: store the current points
835function STORE_Callback(hObject, eventdata, handles)
836%------------------------------------------------------------------------
837Coord=get(handles.ListCoord,'Data');
838unitlist=get(handles.CoordUnit,'String');
839unit=unitlist{get(handles.CoordUnit,'value')};
840GeometryCalib.CoordUnit=unit;
841GeometryCalib.SourceCalib.PointCoord=Coord(:,1:5);
842huvmat=findobj(allchild(0),'Name','uvmat');
843hhuvmat=guidata(huvmat);%handles of elements in the GUI uvmat
844if ~isempty(hhuvmat.RootPath)&& ~isempty(hhuvmat.SubDir)
845    RootPath=get(hhuvmat.RootPath,'String');
846    SubDir=get(hhuvmat.SubDir,'String');
847    filebase=[fullfile(RootPath,SubDir) '~'];
848    while exist([filebase '.xml'],'file')
849        filebase=[filebase '~'];
850    end
851    outputfile=[filebase '.xml'];
852 %   [~,RootFile]=fileparts(filebase);
853
854    t=xmltree;
855    t=set(t,1,'name','ImaDoc');
856    [t,uid_calib]=add(t,1,'element','GeometryCalib');
857   % xmlfile=[filebase '.xml'];
858    t=struct2xml(GeometryCalib,t,uid_calib);
859    save(t)
860    try
861        save(t,outputfile);
862    catch ME
863        errormsg=['error in saving ' outputfile ': ' ME.message];
864         msgbox_uvmat('ERROR',errormsg);
865    end
866    listfile=get(handles.ListCoordFiles,'string');
867    if isequal(listfile,{''})
868        listfile={outputfile};
869    else
870        listfile=[listfile;{outputfile}];%update the list of coord files
871    end
872    set(handles.ListCoordFiles,'string',listfile);
873end
874ClearAll_Callback(hObject, eventdata, handles)% clear the current list and point plots
875
876% --------------------------------------------------------------------
877% --- Executes on button press in ClearAll: clear the list of calibration points
878function ClearAll_Callback(hObject, eventdata, handles)
879% --------------------------------------------------------------------
880set(handles.ListCoord,'Data',[])
881PLOT_Callback(hObject, eventdata, handles)
882update_calib_marker([]);%remove circle marker
883set(handles.APPLY,'backgroundColor',[1 0 0])% set APPLY button to red color: no calibration wanted without points
884set(handles.CheckEnableMouse,'value',1); %activate mouse to pick new points
885
886%------------------------------------------------------------------------
887% --- Executes on button press in CLEAR LIST
888function CLEAR_Callback(hObject, eventdata, handles)
889%------------------------------------------------------------------------
890set(handles.ListCoordFiles,'Value',1)
891set(handles.ListCoordFiles,'String',{''})
892
893%------------------------------------------------------------------------
894% --- Executes on selection change in CheckEnableMouse.
895function CheckEnableMouse_Callback(hObject, eventdata, handles)
896%------------------------------------------------------------------------
897choice=get(handles.CheckEnableMouse,'Value');
898if choice% if button selected
899    set(handles.CheckEnableMouse,'BackgroundColor',[1 1 0])
900    huvmat=findobj(allchild(0),'tag','uvmat');
901    if ishandle(huvmat)
902        hhuvmat=guidata(huvmat);
903        set(hhuvmat.MenuRuler,'checked','off')%desactivate ruler
904        if get(hhuvmat.CheckEditObject,'Value')
905            set(hhuvmat.CheckEditObject,'Value',0)
906            uvmat('CheckEditObject_Callback',hhuvmat.CheckEditObject,[],hhuvmat)
907        end
908        set(hhuvmat.MenuRuler,'checked','off')%desactivate ruler
909    end
910    if ~isequal(get(hhuvmat.TransformName,'Value'),1) %active if transform function is activated to return to the raw image
911        set(hhuvmat.TransformName,'Value',1)
912        uvmat('TransformName_Callback',hObject,eventdata,hhuvmat); %file input with xml reading  in uvmat
913        set(hhuvmat.CheckFixLimits,'Value',0)% put FixedLimits option to 'off' (to sse the whole field)
914         PLOT_Callback(hObject, eventdata, handles)
915    end
916else
917      set(handles.CheckEnableMouse,'BackgroundColor',[0.7 0.7 0.7])
918end
919
920% --------------------------------------------------------------------
921
922% function MenuHelp_Callback(hObject, eventdata, handles)
923% web('http://servforge.legi.grenoble-inp.fr/projects/soft-uvmat/wiki/UvmatHelp#GeometryCalib')
924
925%
926function MenuSetScale_Callback(hObject,eventdata,handles)
927
928answer=msgbox_uvmat('INPUT_TXT','scale pixel/cm?','');
929%create test points
930huvmat=findobj(allchild(0),'Name','uvmat');%find the current uvmat interface handle
931hhuvmat=guidata(huvmat);
932if ~strcmp(get(hhuvmat.Scalar,'Visible'),'on')
933    msgbox_uvmat('ERROR','An image needs to be opened in uvmat for calibration')
934    return
935end
936set(handles.calib_type,'Value',1)% set calib option to 'rescale'
937npx=str2num(get(hhuvmat.num_Npx,'String'))/2;
938npy=str2num(get(hhuvmat.num_Npy,'String'))/2;
939Xima=[0.25*npx 0.75*npx 0.75*npx 0.25*npx]';
940Yima=[0.25*npy 0.25*npy 0.75*npy 0.75*npy]';
941x=Xima/str2num(answer);
942y=Yima/str2num(answer);
943Coord=[x y zeros(4,1) Xima Yima zeros(4,1)];
944set(handles.ListCoord,'Data',Coord)
945set(handles.APPLY,'BackgroundColor',[1 0 1])
946set(handles.CheckEnableMouse,'value',0); %desactivate mouse to avoid spurious points
947
948%------------------------------------------------------------------------
949function MenuCreateGrid_Callback(hObject, eventdata, handles)
950%------------------------------------------------------------------------
951CalibData=get(handles.geometry_calib,'UserData');
952Tinput=[];%default
953if isfield(CalibData,'grid')
954    Tinput=CalibData.grid;
955end
956[T,CalibData.grid]=create_grid(Tinput);%display the GUI create_grid
957set(handles.geometry_calib,'UserData',CalibData)
958
959%grid in phys space
960Coord=get(handles.ListCoord,'Data');
961Coord(1:size(T,1),1:3)=T;%update the existing list of phys coordinates from the GUI create_grid
962set(handles.ListCoord,'Data',Coord)
963set(handles.APPLY,'BackgroundColor',[1 0 1])
964set(handles.CheckEnableMouse,'value',0); %desactivate mouse to avoid spurious points
965
966% -----------------------------------------------------------------------
967% --- automatic grid dectection from local maxima of the images
968function MenuDetectGrid_Callback(hObject, eventdata, handles)
969%------------------------------------------------------------------------
970%% read the four last point coordinates in pixels
971Coord=get(handles.ListCoord,'Data');%read list of coordinates on geometry_calib
972nbpoints=size(Coord,1); %nbre of calibration points
973if nbpoints~=4
974    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')
975    return
976end
977% corners_X=(Coord(end:-1:end-3,4)); %pixel absissa of the four corners
978% corners_Y=(Coord(end:-1:end-3,5));
979corners_X=(Coord(:,4)); %pixel absissa of the four corners
980corners_Y=(Coord(:,5));
981
982%%%%%%
983%   corners_X=1000*[1.5415  1.7557 1.7539 1.5415]';
984%   corners_Y=1000*[1.1515 1.1509 1.3645  1.3639]';
985
986%reorder the last two points (the two first in the list) if needed
987angles=angle((corners_X-corners_X(1))+1i*(corners_Y-corners_Y(1)));
988if abs(angles(4)-angles(2))>abs(angles(3)-angles(2))
989    X_end=corners_X(4);
990    Y_end=corners_Y(4);
991    corners_X(4)=corners_X(3);
992    corners_Y(4)=corners_Y(3);
993    corners_X(3)=X_end;
994    corners_Y(3)=Y_end;
995end
996
997
998
999%% read the current image, displayed in the GUI uvmat
1000huvmat=findobj(allchild(0),'Name','uvmat');
1001UvData=get(huvmat,'UserData');
1002A=UvData.Field.A;%currently displayed image
1003npxy=size(A);
1004
1005%% initiate the grid in phys coordinates
1006CalibData=get(handles.geometry_calib,'UserData');%get information stored on the GUI geometry_calib
1007grid_input=[];%default
1008if isfield(CalibData,'grid')
1009    grid_input=CalibData.grid;%retrieve the previously used grid
1010else
1011   %S=skewness(double(reshape(A,1,[])));
1012   A=double(sum(A,3));%sum of ther three components for color images
1013   A=A-mean(mean(A));
1014   S=mean(mean(A.*A.*A))/(mean(mean(A.*A)))^1.5; %skewness used to istinguish white or black markers
1015   grid_input.CheckWhite=sign(S);%propose white markers if image skewness>0, black markers otherwise
1016end
1017[T,CalibData.grid,CalibData.grid.CheckWhite,CalibData.grid.FilterWindow]=create_grid(grid_input,'detect_grid');%display the GUI create_grid, read the set of phys coordinates T
1018set(handles.geometry_calib,'UserData',CalibData)%store the phys grid parameters for later use
1019X=[CalibData.grid.x_0 CalibData.grid.x_1 CalibData.grid.x_0 CalibData.grid.x_1]';%corner absissa in the phys coordinates (cm)
1020Y=[CalibData.grid.y_0 CalibData.grid.y_0 CalibData.grid.y_1 CalibData.grid.y_1]';%corner ordinates in the phys coordinates (cm)
1021
1022%% calculate transform matrices for plane projection: rectangle assumed to be viewed in perspective
1023% reference: http://alumni.media.mit.edu/~cwren/interpolator/ by Christopher R. Wren
1024B = [ X Y ones(size(X)) zeros(4,3)        -X.*corners_X -Y.*corners_X ...
1025    zeros(4,3)        X Y ones(size(X)) -X.*corners_Y -Y.*corners_Y ];
1026B = reshape (B', 8 , 8 )';
1027D = [ corners_X , corners_Y ];
1028D = reshape (D', 8 , 1 );
1029l = (B' * B)\B' * D;
1030Amat = reshape([l(1:6)' 0 0 1 ],3,3)';
1031C = [l(7:8)' 1];
1032
1033%% transform grid image into 'phys' coordinates
1034GeometryCalib.CalibrationType='3D_linear';
1035GeometryCalib.fx_fy=[1 1];
1036GeometryCalib.Tx_Ty_Tz=[Amat(1,3) Amat(2,3) 1];
1037GeometryCalib.R=[Amat(1,1),Amat(1,2),0;Amat(2,1),Amat(2,2),0;C(1),C(2),0];
1038GeometryCalib.CoordUnit='cm';
1039path_uvmat=which('uvmat');% check the path detected for source file uvmat
1040path_UVMAT=fileparts(path_uvmat); %path to UVMAT
1041addpath(fullfile(path_UVMAT,'transform_field'))
1042Data.ListVarName={'Coord_y','Coord_x','A'};
1043Data.VarDimName={'Coord_y','Coord_x',{'Coord_y','Coord_x'}};
1044if ndims(A)==3
1045    A=mean(A,3);
1046end
1047Data.A=A-min(min(A));
1048Data.Coord_y=[npxy(1)-0.5 0.5];
1049Data.Coord_x=[0.5 npxy(2)];
1050Data.CoordUnit='pixel';
1051Calib.GeometryCalib=GeometryCalib;
1052DataOut=phys(Data,Calib);
1053rmpath(fullfile(path_UVMAT,'transform_field'))
1054Amod=DataOut.A;% current image expressed in 'phys' coord
1055Rangx=DataOut.Coord_x;% x coordinates of first and last pixel centres in phys
1056Rangy=DataOut.Coord_y;% y coordinates of first and last pixel centres in phys
1057
1058
1059%% inverse the image in the case of black lines
1060if CalibData.grid.CheckWhite
1061    Amod=double(Amod);%case of white grid markers: will look for image maxima
1062else
1063    Amod=-double(Amod);%case of black grid markers: will look for image minima
1064end
1065
1066%%%%%%filterfor i;proved detection of dots
1067if ~isequal(CalibData.grid.FilterWindow,0)
1068    %definition of the cos shape matrix filter
1069    FilterBoxSize_x=CalibData.grid.FilterWindow;
1070    FilterBoxSize_y=CalibData.grid.FilterWindow;
1071    ix=1/2-FilterBoxSize_x/2:-1/2+FilterBoxSize_x/2;%
1072    iy=1/2-FilterBoxSize_y/2:-1/2+FilterBoxSize_y/2;%
1073    %del=np/3;
1074    %fct=exp(-(ix/del).^2);
1075    fct2_x=cos(ix/((FilterBoxSize_x-1)/2)*pi/2);
1076    fct2_y=cos(iy/((FilterBoxSize_y-1)/2)*pi/2);
1077    %Mfiltre=(ones(5,5)/5^2);
1078    Mfiltre=fct2_y'*fct2_x;
1079    Mfiltre=Mfiltre/(sum(sum(Mfiltre)));%normalize filter
1080   
1081    if ndims(Amod)==3
1082        Amod=filter2(Mfiltre,sum(Amod,3));%filter the input image, after summation on the color component (for color images)
1083    else
1084        Amod=filter2(Mfiltre,Amod);
1085    end
1086end
1087%%%%%%%%%%%%%%
1088
1089
1090%% detection of local image extrema in each direction
1091Dx=(Rangx(2)-Rangx(1))/(npxy(2)-1); %x mesh in real space
1092Dy=(Rangy(2)-Rangy(1))/(npxy(1)-1); %y mesh in real space
1093ind_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
1094ind_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
1095nbpoints=size(T,1);
1096TIndex=ones(size(T));% image indices corresponding to point coordinates
1097%look for image maxima around each expected grid point
1098for ipoint=1:nbpoints
1099    i0=1+round((T(ipoint,1)-Rangx(1))/Dx);% x index of the expected point in the phys image Amod
1100    j0=1+round((T(ipoint,2)-Rangy(1))/Dy);% y index of the expected point in the phys image Amod
1101    j0min=max(j0-ind_range_y,1);% min y index selected for the subimage (cut at the edge to avoid index <1)
1102    j0max=min(j0+ind_range_y,size(Amod,1));% max y index selected for the subimage (cut at the edge to avoid index > size)
1103    i0min=max(i0-ind_range_x,1);% min x index selected for the subimage (cut at the edge to avoid index <1)
1104    i0max=min(i0+ind_range_x,size(Amod,2));% max x index selected for the subimage (cut at the edge to avoid index > size)
1105    Asub=Amod(j0min:j0max,i0min:i0max); %subimage used to find brigthness extremum
1106    x_profile=sum(Asub,1);%profile of subimage summed over y
1107    y_profile=sum(Asub,2);%profile of subimage summed over x
1108   
1109    [tild,ind_x_max]=max(x_profile);% index of max for the x profile
1110    [tild,ind_y_max]=max(y_profile);% index of max for the y profile
1111    %sub-pixel improvement using moments
1112    x_shift=0;
1113    y_shift=0;
1114    if ind_x_max+2<=numel(x_profile) && ind_x_max-2>=1
1115        Atop=x_profile(ind_x_max-2:ind_x_max+2);% extract x profile around the max
1116        x_shift=sum(Atop.*[-2 -1 0 1 2])/sum(Atop);
1117    end
1118    if ind_y_max+2<=numel(y_profile) && ind_y_max-2>=1
1119        Atop=y_profile(ind_y_max-2:ind_y_max+2);% extract y profile around the max
1120        y_shift=sum(Atop.*[-2 -1 0 1 2]')/sum(Atop);
1121    end
1122    %%%%
1123    %     if ipoint==9
1124    %                 figure(11)
1125    %   imagesc(Asub)
1126    %     figure(12)
1127    %     plot(x_profile,'r')
1128    %     hold on
1129    %     plot(y_profile,'b')
1130    %     grid on
1131    %     end
1132    %%%%
1133    TIndex(ipoint,1)=(i0min+ind_x_max-1+x_shift);% x position of the maximum (in index of Amod)
1134    TIndex(ipoint,2)=(j0min+ind_y_max-1+y_shift);% y position of the maximum (in index of Amod)
1135end
1136Tmod(:,1)=(TIndex(:,1)-1)*Dx+Rangx(1);
1137Tmod(:,2)=(TIndex(:,2)-1)*Dy+Rangy(1);
1138%Tmod=T(:,(1:2))+Delta;% 'phys' coordinates of the detected points
1139[Xpx,Ypx]=px_XYZ(GeometryCalib,[],Tmod(:,1),Tmod(:,2));% image coordinates of the detected points
1140Coord=[T Xpx Ypx zeros(size(T,1),1)];
1141set(handles.ListCoord,'Data',Coord)
1142PLOT_Callback(hObject, eventdata, handles)
1143set(handles.APPLY,'BackgroundColor',[1 0 1])
1144set(handles.CheckEnableMouse,'value',0); %desactivate mouse to avoid spurious points
1145
1146
1147%-----------------------------------------------------------------------
1148function MenuTranslatePoints_Callback(hObject, eventdata, handles)
1149%-----------------------------------------------------------------------
1150%hcalib=get(handles.calib_type,'parent');%handles of the GUI geometry_calib
1151CalibData=get(handles.geometry_calib,'UserData');
1152Tinput=[];%default
1153if isfield(CalibData,'translate')
1154    Tinput=CalibData.translate;
1155end
1156T=translate_points(Tinput);%display translate_points GUI and get shift parameters
1157CalibData.translate=T;
1158set(handles.geometry_calib,'UserData',CalibData)
1159%translation
1160Coord=get(handles.ListCoord,'Data');
1161Coord(:,1)=T(1)+Coord(:,1);
1162Coord(:,2)=T(2)+Coord(:,2);
1163Coord(:,3)=T(3)+Coord(:,3);
1164set(handles.ListCoord,'Data',Coord);
1165set(handles.APPLY,'BackgroundColor',[1 0 1])
1166
1167% --------------------------------------------------------------------
1168function MenuRotatePoints_Callback(hObject, eventdata, handles)
1169%hcalib=get(handles.calib_type,'parent');%handles of the GUI geometry_calib
1170CalibData=get(handles.geometry_calib,'UserData');
1171Tinput=[];%default
1172if isfield(CalibData,'rotate')
1173    Tinput=CalibData.rotate;
1174end
1175T=rotate_points(Tinput);%display rotate_points GUI to introduce rotation parameters
1176CalibData.rotate=T;
1177set(handles.geometry_calib,'UserData',CalibData)
1178%-----------------------------------------------------
1179%rotation
1180Phi=T(1);
1181O_x=0;%default
1182O_y=0;%default
1183if numel(T)>=2
1184    O_x=T(2);%default
1185end
1186if numel(T)>=3
1187    O_y=T(3);%default
1188end
1189Coord=get(handles.ListCoord,'Data');
1190r1=cos(pi*Phi/180);
1191r2=-sin(pi*Phi/180);
1192r3=sin(pi*Phi/180);
1193r4=cos(pi*Phi/180);
1194x=Coord(:,1)-O_x;
1195y=Coord(:,2)-O_y;
1196Coord(:,1)=r1*x+r2*y;
1197Coord(:,2)=r3*x+r4*y;
1198set(handles.ListCoord,'Data',Coord)
1199set(handles.APPLY,'BackgroundColor',[1 0 1])
1200
1201% --------------------------------------------------------------------
1202function MenuFlip_xy_Callback(hObject, eventdata, handles)
1203Coord=get(handles.ListCoord,'Data');
1204Coord(:,1)=-Coord(:,1);
1205Coord(:,2)=-Coord(:,2);
1206set(handles.ListCoord,'Data',Coord)
1207
1208% --------------------------------------------------------------------
1209function MenuFlip_xz_Callback(hObject, eventdata, handles)
1210Coord=get(handles.ListCoord,'Data');
1211Coord(:,1)=-Coord(:,1);
1212Coord(:,3)=-Coord(:,3);
1213set(handles.ListCoord,'Data',Coord)
1214
1215% --------------------------------------------------------------------
1216function MenuImportPoints_Callback(hObject, eventdata, handles)
1217fileinput=browse_xml(hObject, eventdata, handles);
1218if isempty(fileinput)
1219    return
1220end
1221[s,errormsg]=imadoc2struct(fileinput,'GeometryCalib');
1222if ~isfield(s,'GeometryCalib')
1223    msgbox_uvmat('ERROR','invalid input file: no geometry_calib data')
1224    return
1225end
1226GeometryCalib=s.GeometryCalib;
1227if ~(isfield(GeometryCalib,'SourceCalib')&&isfield(GeometryCalib.SourceCalib,'PointCoord'))
1228    msgbox_uvmat('ERROR','invalid input file: no calibration points')
1229    return
1230end
1231Coord=GeometryCalib.SourceCalib.PointCoord;
1232Coord=[Coord zeros(size(Coord,1),1)];
1233set(handles.ListCoord,'Data',Coord)
1234PLOT_Callback(handles.geometry_calib, [], handles)
1235set(handles.APPLY,'BackgroundColor',[1 0 1])
1236set(handles.CheckEnableMouse,'value',0); %desactivate mouse to avoid modifications by default
1237
1238% -----------------------------------------------------------------------
1239function MenuImportIntrinsic_Callback(hObject, eventdata, handles)
1240%------------------------------------------------------------------------
1241fileinput=browse_xml(hObject, eventdata, handles);
1242if isempty(fileinput)
1243    return
1244end
1245[s,errormsg]=imadoc2struct(fileinput,'GeometryCalib');
1246GeometryCalib=s.GeometryCalib;
1247display_intrinsic(GeometryCalib,handles)
1248
1249% -----------------------------------------------------------------------
1250function MenuImportAll_Callback(hObject, eventdata, handles)
1251%------------------------------------------------------------------------
1252fileinput=browse_xml(hObject, eventdata, handles);
1253if ~isempty(fileinput)
1254    loadfile(handles,fileinput)
1255end
1256set(handles.CheckEnableMouse,'value',0); %desactivate mouse to avoid modifications by default
1257
1258% -----------------------------------------------------------------------
1259% --- Executes on menubar option Import/Grid file: introduce previous grid files
1260function MenuGridFile_Callback(hObject, eventdata, handles)
1261% -----------------------------------------------------------------------
1262inputfile=browse_xml(hObject, eventdata, handles);
1263listfile=get(handles.ListCoordFiles,'String');
1264if isequal(listfile,{''})
1265    listfile={inputfile};
1266else
1267    listfile=[listfile;{inputfile}];%update the list of coord files
1268end
1269set(handles.ListCoordFiles,'String',listfile);
1270
1271
1272%------------------------------------------------------------------------
1273function fileinput=browse_xml(hObject, eventdata, handles)
1274%------------------------------------------------------------------------
1275fileinput=[];%default
1276oldfile=''; %default
1277UserData=get(handles.geometry_calib,'UserData');
1278if isfield(UserData,'XmlInputFile')
1279    oldfile=UserData.XmlInputFile;
1280end
1281[FileName, PathName, filterindex] = uigetfile( ...
1282    {'*.xml;*.mat', ' (*.xml,*.mat)';
1283    '*.xml',  '.xml files '; ...
1284    '*.mat',  '.mat matlab files '}, ...
1285    'Pick a file',oldfile);
1286fileinput=[PathName FileName];%complete file name
1287testblank=findstr(fileinput,' ');%look for blanks
1288if ~isempty(testblank)
1289    msgbox_uvmat('ERROR','forbidden input file name or path: no blank character allowed')
1290    return
1291end
1292sizf=size(fileinput);
1293if (~ischar(fileinput)||~isequal(sizf(1),1)),return;end
1294UserData.XmlInputFile=fileinput;
1295set(handles.geometry_calib,'UserData',UserData)%record current file foer further use of browser
1296
1297% -----------------------------------------------------------------------
1298function Heading=loadfile(handles,fileinput)
1299%------------------------------------------------------------------------
1300Heading=[];%default
1301[s,errormsg]=imadoc2struct(fileinput,'Heading','GeometryCalib');
1302if ~isempty(errormsg)
1303    msgbox_uvmat('ERROR',errormsg)
1304    return
1305end
1306if ~isempty(s.Heading)
1307    Heading=s.Heading;
1308end
1309checkcoord=false;
1310if isfield (s,'GeometryCalib')
1311    GeometryCalib=s.GeometryCalib;
1312    %fx=1;fy=1;Cx=0;Cy=0;kc=0; %default
1313    %CoordCell={};
1314    %Tabchar={};%default
1315    val_cal=1;%default
1316    if ~isempty(GeometryCalib)
1317        % choose the calibration option
1318        if isfield(GeometryCalib,'CalibrationType')
1319            calib_list=get(handles.calib_type,'String');
1320            for ilist=1:numel(calib_list)
1321                if strcmp(calib_list{ilist},GeometryCalib.CalibrationType)
1322                    val_cal=ilist;
1323                    break
1324                end
1325            end
1326        end
1327        display_intrinsic(GeometryCalib,handles)%intrinsic param
1328        %extrinsic param
1329        if isfield(GeometryCalib,'Tx_Ty_Tz')
1330            %Tx_Ty_Tz=GeometryCalib.Tx_Ty_Tz;
1331            set(handles.Tx,'String',num2str(GeometryCalib.Tx_Ty_Tz(1),4))
1332            set(handles.Ty,'String',num2str(GeometryCalib.Tx_Ty_Tz(2),4))
1333            set(handles.Tz,'String',num2str(GeometryCalib.Tx_Ty_Tz(3),4))
1334        end
1335        if isfield(GeometryCalib,'omc')
1336            set(handles.Phi,'String',num2str(GeometryCalib.omc(1),4))
1337            set(handles.Theta,'String',num2str(GeometryCalib.omc(2),4))
1338            set(handles.Psi,'String',num2str(GeometryCalib.omc(3),4))
1339        end
1340        if isfield(GeometryCalib,'SourceCalib')&& isfield(GeometryCalib.SourceCalib,'PointCoord')
1341            calib=GeometryCalib.SourceCalib.PointCoord;
1342            Coord=[calib zeros(size(calib,1),1)];
1343            set(handles.ListCoord,'Data',Coord)
1344            checkcoord=true;
1345        end
1346        PLOT_Callback(handles.geometry_calib, [], handles)
1347        set(handles.APPLY,'BackgroundColor',[1 0 1])
1348    end
1349    set(handles.calib_type,'Value',val_cal)
1350end   
1351if checkcoord % does not allow mouse action by default in the presence of input points
1352    set(handles.CheckEnableMouse,'Value',0)
1353    set(handles.CheckEnableMouse,'BackgroundColor',[0.7 0.7 0.7])
1354else   % allow mouse action by default in the absence of input points
1355    set(handles.CheckEnableMouse,'Value',1)
1356    set(handles.CheckEnableMouse,'BackgroundColor',[1 1 0])
1357end
1358
1359%------------------------------------------------------------------------
1360%---display calibration intrinsic parameters
1361function display_intrinsic(GeometryCalib,handles)
1362%------------------------------------------------------------------------
1363fx=[];
1364fy=[];
1365if isfield(GeometryCalib,'fx_fy')
1366    fx=GeometryCalib.fx_fy(1);
1367    fy=GeometryCalib.fx_fy(2);
1368end
1369Cx_Cy=[0 0];%default
1370if isfield(GeometryCalib,'Cx_Cy')
1371    Cx_Cy=GeometryCalib.Cx_Cy;
1372end
1373kc=0;
1374if isfield(GeometryCalib,'kc')
1375    kc=GeometryCalib.kc; %* GeometryCalib.focal*GeometryCalib.focal;
1376end
1377set(handles.fx,'String',num2str(fx,5))
1378set(handles.fy,'String',num2str(fy,5))
1379set(handles.Cx,'String',num2str(Cx_Cy(1),'%1.1f'))
1380set(handles.Cy,'String',num2str(Cx_Cy(2),'%1.1f'))
1381%set(handles.kc,'String',num2str(kc,'%1.4f'))
1382set(handles.kc,'String',num2str(kc,4))
1383
1384%------------------------------------------------------------------------
1385% ---display calibration extrinsic parameters
1386function display_extrinsic(GeometryCalib,handles)
1387%------------------------------------------------------------------------
1388set(handles.Tx,'String',num2str(GeometryCalib.Tx_Ty_Tz(1),4))
1389set(handles.Ty,'String',num2str(GeometryCalib.Tx_Ty_Tz(2),4))
1390set(handles.Tz,'String',num2str(GeometryCalib.Tx_Ty_Tz(3),4))
1391set(handles.Phi,'String',num2str(GeometryCalib.omc(1),4))
1392set(handles.Theta,'String',num2str(GeometryCalib.omc(2),4))
1393set(handles.Psi,'String',num2str(GeometryCalib.omc(3),4))
1394
1395%------------------------------------------------------------------------
1396% --- executes when user attempts to close geometry_calib.
1397function geometry_calib_CloseRequestFcn(hObject, eventdata, handles)
1398%------------------------------------------------------------------------
1399delete(hObject); % closes the figure
1400
1401%------------------------------------------------------------------------
1402% --- executes on button press in PLOT.
1403%------------------------------------------------------------------------
1404function PLOT_Callback(hObject, eventdata, handles)
1405huvmat=findobj(allchild(0),'Name','uvmat');%find the current uvmat interface handle
1406hhuvmat=guidata(huvmat); %handles of GUI elements in uvmat
1407h_menu_coord=findobj(huvmat,'Tag','TransformName');
1408menu=get(h_menu_coord,'String');
1409choice=get(h_menu_coord,'Value');
1410option='';
1411if iscell(menu)
1412    option=menu{choice};
1413end
1414Coord=get(handles.ListCoord,'Data');
1415if ~isempty(Coord)
1416    if isequal(option,'phys')
1417        Coord_plot=Coord(:,1:3);
1418    elseif isempty(option);%optionoption,'px')||isequal(option,'')
1419        Coord_plot=Coord(:,4:5);
1420    else
1421        msgbox_uvmat('ERROR','the choice in menu_coord of uvmat must be blank or phys ')
1422    end
1423end
1424
1425set(0,'CurrentFigure',huvmat)
1426set(huvmat,'CurrentAxes',hhuvmat.PlotAxes)
1427hh=findobj('Tag','calib_points');
1428if  ~isempty(Coord) && isempty(hh)
1429    hh=line(Coord_plot(:,1),Coord_plot(:,2),'Color','m','Tag','calib_points','LineStyle','none','Marker','+','MarkerSize',10);
1430elseif isempty(Coord)%empty list of points, suppress the plot
1431    delete(hh)
1432else
1433    set(hh,'XData',Coord_plot(:,1))
1434    set(hh,'YData',Coord_plot(:,2))
1435end
1436pause(.1)
1437figure(handles.geometry_calib)
1438
1439%------------------------------------------------------------------------
1440% --- Executes on button press in Copy: display Coord on the Matlab work space
1441%------------------------------------------------------------------------
1442function Copy_Callback(hObject, eventdata, handles)
1443global Coord
1444if isequal (get(handles.Copy,'BackGroundColor'),[0 1 0])
1445    evalin('base','global Coord')%make CurData global in the workspace
1446    Coord=get(handles.ListCoord,'Data');
1447    display('coordinates of calibration points (phys,px,marker) :')
1448    evalin('base','Coord') %display CurData in the workspace
1449    commandwindow; %brings the Matlab command window to the front
1450    set(handles.Copy,'BackGroundColor',[1 1 0])
1451    set(handles.Copy,'String','IMPORT PTS')
1452else
1453    set(handles.ListCoord,'Data',Coord)
1454    set(handles.Copy,'String','EXPORT PTS')
1455    set(handles.Copy,'BackGroundColor',[0 1 0])
1456end
1457
1458%------------------------------------------------------------------------
1459% --- Executes when selected cell(s) is changed in ListCoord.
1460%------------------------------------------------------------------------
1461function ListCoord_CellSelectionCallback(hObject, eventdata, handles)
1462if ~isempty(eventdata.Indices)
1463    iline=eventdata.Indices(1);% selected line number
1464    set(handles.CoordLine,'String',num2str(iline))
1465    Data=get(handles.ListCoord,'Data');
1466    update_calib_marker(Data(iline,:))
1467end
1468
1469%------------------------------------------------------------------------
1470% --- Executes when entered data in editable cell(s) in ListCoord.
1471%------------------------------------------------------------------------
1472function ListCoord_CellEditCallback(hObject, eventdata, handles)
1473
1474Input=str2num(eventdata.EditData);%pasted input
1475Coord=get(handles.ListCoord,'Data');
1476iline=eventdata.Indices(1);% selected line number
1477if numel(Input)>1 && size(Coord,1)<iline+numel(Input)
1478    Coord=[Coord ; zeros(iline+numel(Input)-size(Coord,1),6)];% append zeros to fit the new column
1479end
1480Coord(iline:iline+numel(Input)-1,eventdata.Indices(2))=Input';
1481set(handles.ListCoord,'Data',Coord)
1482PLOT_Callback(hObject, eventdata, handles)
1483
1484%------------------------------------------------------------------------
1485% --- 'key_press_fcn:' function activated when a key is pressed on the keyboard
1486%------------------------------------------------------------------------
1487function ListCoord_KeyPressFcn(hObject, eventdata, handles)
1488iline=str2num(get(handles.CoordLine,'String'));
1489xx=double(get(handles.geometry_calib,'CurrentCharacter'));%get the keyboard character
1490if ismember(xx,[28 29 30 31])% directional arrow
1491    Coord=get(handles.ListCoord,'Data');
1492    switch xx
1493        case 30 % arrow upward
1494            iline=iline-1;
1495        case 31% arrow downward
1496            iline=iline+1;
1497    end
1498    if iline>=1 && iline<=size(Coord,1)
1499        set(handles.CoordLine,'String',num2str(iline))
1500        update_calib_marker(Coord(iline,1:5))% show the point corresponding to the selected line
1501    end
1502else
1503    set(handles.APPLY,'BackgroundColor',[1 0 1])% paint APPLY in magenta to indicate that the table content has be modified
1504    if ismember(xx,[127 31])% delete, or downward
1505        Coord=get(handles.ListCoord,'Data');
1506        iline=str2double(get(handles.CoordLine,'String'));
1507        if isequal(xx, 31)
1508            if isequal(iline,size(Coord,1))% arrow downward
1509                Coord=[Coord;zeros(1,size(Coord,2))];
1510            end
1511        else
1512            Coord(iline,:)=[];% suppress the current line
1513        end
1514        set(handles.ListCoord,'Data',Coord);
1515    end
1516end
1517
1518%------------------------------------------------------------------------
1519% --- update the plot of calibration points
1520%------------------------------------------------------------------------
1521% draw a circle around the point defined by the input coordinates Coord as given by line in the table Listcoord
1522function update_calib_marker(Coord)
1523
1524%% read config on uvmat
1525huvmat=findobj(allchild(0),'Name','uvmat');%find the current uvmat interface handle
1526hhuvmat=guidata(huvmat);
1527hhh=findobj(hhuvmat.PlotAxes,'Tag','calib_marker');
1528if numel(Coord)<5
1529    if ~isempty(hhh)
1530        delete(hhh)%delete the circle marker in case of no valid input
1531    end
1532    return
1533end
1534menu=get(hhuvmat.TransformName,'String');
1535choice=get(hhuvmat.TransformName,'Value');
1536option='';
1537if iscell(menu)
1538    option=menu{choice};
1539end
1540
1541%% read appropriate coordinates (px or phys) in the table ListCoord
1542if isequal(option,'phys') % use phys coord
1543    XCoord=Coord(1);
1544    YCoord=Coord(2);
1545elseif isempty(option)% use coord in pixels
1546    XCoord=Coord(4);
1547    YCoord=Coord(5);
1548else
1549    msgbox_uvmat('ERROR','the choice in menu_coord of uvmat must be blank or phys ')
1550    return
1551end
1552
1553%% adjust the plot limits if needed
1554xlim=get(hhuvmat.PlotAxes,'XLim');
1555ylim=get(hhuvmat.PlotAxes,'YLim');
1556ind_range=max(abs(xlim(2)-xlim(1)),abs(ylim(end)-ylim(1)))/25;%defines the size of the circle marker
1557check_xlim=0;
1558if XCoord>xlim(2)
1559    xlim=xlim+XCoord-xlim(2)+ind_range;% translate plot limit
1560    check_xlim=1;
1561elseif XCoord<xlim(1)
1562    xlim=xlim-XCoord+xlim(1)-ind_range;% translate plot limit
1563    check_xlim=1;
1564end
1565if check_xlim
1566    set(hhuvmat.PlotAxes,'XLim',xlim);
1567    set(hhuvmat.num_MaxX,'String',num2str(xlim(2)));
1568    set(hhuvmat.num_MinX,'String',num2str(xlim(1)));
1569end
1570check_ylim=0;
1571if YCoord>ylim(2)
1572    ylim=ylim+YCoord-ylim(2)+ind_range;% translate plot limit
1573    check_ylim=1;
1574elseif YCoord<ylim(1)
1575    ylim=ylim-YCoord+ylim(1)-ind_range;% translate plot limit
1576    check_ylim=1;
1577end
1578if check_ylim
1579    set(hhuvmat.PlotAxes,'YLim',ylim);
1580    set(hhuvmat.num_MaxY,'String',num2str(ylim(2)));
1581    set(hhuvmat.num_MinY,'String',num2str(ylim(1)));
1582end
1583
1584%% plot a circle around the selected point
1585if isempty(hhh)
1586    set(0,'CurrentFig',huvmat)
1587    set(huvmat,'CurrentAxes',hhuvmat.PlotAxes)
1588    rectangle('Curvature',[1 1],...
1589        'Position',[XCoord-ind_range/2 YCoord-ind_range/2 ind_range ind_range],'EdgeColor','m',...
1590        'LineStyle','-','Tag','calib_marker');
1591else
1592    set(hhh,'Position',[XCoord-ind_range/2 YCoord-ind_range/2 ind_range ind_range])
1593end
1594
1595%------------------------------------------------------------------------
1596% --- Executes on button press in ClearPoint: remove the selected line in the table Coord
1597%------------------------------------------------------------------------
1598function ClearPoint_Callback(hObject, eventdata, handles)
1599
1600Coord=get(handles.ListCoord,'Data');
1601iline=str2num(get(handles.CoordLine,'String'));
1602if isempty(iline)
1603    msgbox_uvmat('WARNING','no line suppressed, select a line in the table')
1604else
1605    answer=msgbox_uvmat('INPUT_Y-N',['suppress line ' num2str(iline) '?']);
1606    if isequal(answer,'Yes')
1607        Coord(iline,:)=[];
1608        set(handles.APPLY,'BackgroundColor',[1 0 1])
1609        set(handles.ListCoord,'Data',Coord);
1610        set(handles.CoordLine,'String','')
1611        PLOT_Callback(hObject,eventdata,handles)
1612        update_calib_marker([]);%remove circle marker
1613    end
1614end
1615
1616
1617
Note: See TracBrowser for help on using the repository browser.