source: trunk/src/series/aver_stat.m @ 525

Last change on this file since 525 was 494, checked in by sommeria, 12 years ago

various bugs corrected after testing in Windows OS. Introduction
of filter tps

File size: 17.5 KB
RevLine 
[169]1%'aver_stat': calculate field average, used with series.fig
[457]2% this function can be used as a template for applying a global operation (here averaging) on a series of input fields
[169]3%------------------------------------------------------------------------
[457]4% function ParamOut=aver_stat(Param)
[169]5%
[447]6%%%%%%%%%%% GENERAL TO ALL SERIES ACTION FCTS %%%%%%%%%%%%%%%%%%%%%%%%%%%
[457]7%
8% This function is used in four modes by the GUI series:
9%           1) config GUI: with no input argument, the function determine the suitable GUI configuration
10%           2) interactive input: the function is used to interactively introduce input parameters, and then stops
11%           3) RUN: the function itself runs, when an appropriate input  structure Param has been introduced.
12%           4) BATCH: the function itself proceeds in BATCH mode, using an xml file 'Param' as input.
13%
14% This function is used in four modes by the GUI series:
15%           1) config GUI: with no input argument, the function determine the suitable GUI configuration
16%           2) interactive input: the function is used to interactively introduce input parameters, and then stops
17%           3) RUN: the function itself runs, when an appropriate input  structure Param has been introduced.
18%           4) BATCH: the function itself proceeds in BATCH mode, using an xml file 'Param' as input.
19%
[169]20%OUTPUT
21% GUI_input=list of options in the GUI series.fig needed for the function
22%
23%INPUT:
[447]24% In run mode, the input parameters are given as a Matlab structure Param copied from the GUI series.
25% In batch mode, Param is the name of the corresponding xml file containing the same information
26% In the absence of input (as activated when the current Action is selected
27% in series), the function ouput GUI_input set the activation of the needed GUI elements
[169]28%
[447]29% Param contains the elements:(use the menu bar command 'export/GUI config' in series to see the current structure Param)
30%    .InputTable: cell of input file names, (several lines for multiple input)
31%                      each line decomposed as {RootPath,SubDir,Rootfile,NomType,Extension}
32%    .OutputSubDir: name of the subdirectory for data outputs
[474]33%    .OutputDirExt: directory extension for data outputs
[447]34%    .Action: .ActionName: name of the current activated function
35%             .ActionPath:   path of the current activated function
36%    .IndexRange: set the file or frame indices on which the action must be performed
37%    .FieldTransform: .TransformName: name of the selected transform function
38%                     .TransformPath:   path  of the selected transform function
39%                     .TransformHandle: corresponding function handle
40%    .InputFields: sub structure describing the input fields withfields
41%              .FieldName: name of the field
42%              .VelType: velocity type
43%              .FieldName_1: name of the second field in case of two input series
44%              .VelType_1: velocity type of the second field in case of two input series
45%    .ProjObject: %sub structure describing a projection object (read from ancillary GUI set_object)
46%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
47
[457]48function ParamOut=aver_stat(Param)
[447]49
50%% set the input elements needed on the GUI series when the action is selected in the menu ActionName
51if ~exist('Param','var') % case with no input parameter
[474]52    ParamOut={'AllowInputSort';'off';...% allow alphabetic sorting of the list of input files (options 'off'/'on', 'off' by default)
[457]53        'WholeIndexRange';'off';...% prescribes the file index ranges from min to max (options 'off'/'on', 'off' by default)
[27]54        'NbSlice';'on'; ...%nbre of slices ('off' by default)
[447]55        'VelType';'two';...% menu for selecting the velocity type (options 'off'/'one'/'two',  'off' by default)
56        'FieldName';'two';...% menu for selecting the field (s) in the input file(options 'off'/'one'/'two', 'off' by default)
57        'FieldTransform'; 'on';...%can use a transform function
58        'ProjObject';'on';...%can use projection object(option 'off'/'on',
59        'Mask';'off';...%can use mask option   (option 'off'/'on', 'off' by default)
60        'OutputDirExt';'.stat';...%set the output dir extension
[27]61               ''};
62        return
63end
64
[474]65%%%%%%%%%%%%  STANDARD PART  %%%%%%%%%%%%
[457]66%% select different modes,  RUN, parameter input, BATCH
[447]67% BATCH  case: read the xml file for batch case
[457]68if ischar(Param)
69        Param=xml2struct(Param);
70        checkrun=0;
71% RUN case: parameters introduced as the input structure Param
72else
[462]73    if isfield(Param,'Specific')&& strcmp(Param.Specific,'?')
74        checkrun=1;% will only search interactive input parameters (preparation of BATCH mode)
75    else
76        checkrun=2; % indicate the RUN option is used
77    end
[478]78    hseries=guidata(Param.hseries);%handles of the GUI series
[361]79end
[462]80ParamOut=Param; %default output
[474]81OutputDir=[Param.OutputSubDir Param.OutputDirExt];
82   
[457]83%% root input file(s) and type
84RootPath=Param.InputTable(:,1);
85RootFile=Param.InputTable(:,3);
86SubDir=Param.InputTable(:,2);
87NomType=Param.InputTable(:,4);
88FileExt=Param.InputTable(:,5);
[374]89[filecell,i1_series,i2_series,j1_series,j2_series]=get_file_series(Param);
[474]90%%%%%%%%%%%%
91% The cell array filecell is the list of input file names, while
92% filecell{iview,fileindex}:
[447]93%        iview: line in the table corresponding to a given file series
94%        fileindex: file index within  the file series,
95% i1_series(iview,ref_j,ref_i)... are the corresponding arrays of indices i1,i2,j1,j2, depending on the input line iview and the two reference indices ref_i,ref_j
96% i1_series(iview,fileindex) expresses the same indices as a 1D array in file indices
[474]97%%%%%%%%%%%%
[447]98NbSlice=1;%default
[457]99if isfield(Param.IndexRange,'NbSlice')&&~isempty(Param.IndexRange.NbSlice)
[447]100    NbSlice=Param.IndexRange.NbSlice;
[27]101end
[454]102nbview=numel(i1_series);%number of input file series (lines in InputTable)
[457]103nbfield_j=size(i1_series{1},1); %nb of fields for the j index (bursts or volume slices)
104nbfield_i=size(i1_series{1},2); %nb of fields for the i index
105nbfield=nbfield_j*nbfield_i; %total number of fields
[454]106nbfield_i=floor(nbfield/NbSlice);%total number of  indexes in a slice (adjusted to an integer number of slices)
107nbfield=nbfield_i*NbSlice; %total number of fields after adjustement
[43]108
[447]109%determine the file type on each line from the first input file
110ImageTypeOptions={'image','multimage','mmreader','video'};
111NcTypeOptions={'netcdf','civx','civdata'};
[27]112for iview=1:nbview
[451]113    if ~exist(filecell{iview,1}','file')
114        msgbox_uvmat('ERROR',['the first input file ' filecell{iview,1} ' does not exist'])
115        return
116    end
117    [FileType{iview},FileInfo{iview},MovieObject{iview}]=get_file_type(filecell{iview,1});
[447]118    CheckImage{iview}=~isempty(find(strcmp(FileType{iview},ImageTypeOptions)));% =1 for images
119    CheckNc{iview}=~isempty(find(strcmp(FileType{iview},NcTypeOptions)));% =1 for netcdf files
[454]120    if ~isempty(j1_series{iview})
121        frame_index{iview}=j1_series{iview};
122    else
123        frame_index{iview}=i1_series{iview};
124    end
[27]125end
126
[451]127%% calibration data and timing: read the ImaDoc files
[470]128[XmlData,NbSlice_calib,time,errormsg]=read_multimadoc(RootPath,SubDir,RootFile,FileExt,i1_series,i2_series,j1_series,j2_series);
129if size(time,1)>1
[27]130    diff_time=max(max(diff(time)));
131    if diff_time>0
[447]132        msgbox_uvmat('WARNING',['times of series differ by (max) ' num2str(diff_time)])
[27]133    end   
134end
135
[447]136%% coordinate transform or other user defined transform
137transform_fct='';%default
[478]138if isfield(Param,'FieldTransform')&&~isempty(Param.FieldTransform.TransformName)
[474]139    addpath(Param.FieldTransform.TransformPath)
140    transform_fct=str2func(Param.FieldTransform.TransformName);
141    rmpath(Param.FieldTransform.TransformPath)
[27]142end
[474]143
[447]144%%%%%%%%%%%% END STANDARD PART  %%%%%%%%%%%%
145 % EDIT FROM HERE
146
147%% check the validity of  input file types
148if CheckImage{1}
149    FileExtOut='.png'; % write result as .png images for image inputs
150elseif CheckNc{1}
151    FileExtOut='.nc';% write result as .nc files for netcdf inputs
152else
153    msgbox_uvmat('ERROR',['invalid file type input ' FileType{1}])
154    return
155end
156if nbview==2 && ~isequal(CheckImage{1},CheckImage{2})
157        msgbox_uvmat('ERROR','input must be two image series or two netcdf file series')
158    return
159end
160NomTypeOut='_1-2_1';% output file index will indicate the first and last ref index in the series
[474]161if checkrun==1
162    return % stop here for input checks
[218]163end
[442]164
[447]165%% Set field names and velocity types
166InputFields{1}=[];%default (case of images)
167if isfield(Param,'InputFields')
168    InputFields{1}=Param.InputFields;
169end
170if nbview==2
171    InputFields{2}=[];%default (case of images)
172    if isfield(Param,'InputFields')
173        InputFields{2}=Param.InputFields{1};%default
174        if isfield(Param.InputFields,'FieldName_1')
175            InputFields{2}.FieldName=Param.InputFields.FieldName_1;
176            if isfield(Param.InputFields,'VelType_1')
177                InputFields{2}.VelType=Param.InputFields.VelType_1;
178            end
179        end
[255]180    end
181end
[447]182
183%% MAIN LOOP ON SLICES
184%%%%%%%%%%%%% STANDARD PART (DO NOT EDIT) %%%%%%%%%%%%
[27]185for i_slice=1:NbSlice
[447]186    index_slice=i_slice:NbSlice:nbfield;% select file indices of the slice
[255]187    nbfiles=0;
188    nbmissing=0;
[474]189
[447]190    %%%%%%%%%%%%%%%% loop on field indices %%%%%%%%%%%%%%%%
191    for index=index_slice
192        if checkrun
[478]193            update_waitbar(hseries.Waitbar,index/(nbfield))
[447]194            stopstate=get(hseries.RUN,'BusyAction');
195        else
196            stopstate='queue';
197        end
198       
199        %%%%%%%%%%%%%%%% loop on views (input lines) %%%%%%%%%%%%%%%%
200        for iview=1:nbview
[255]201            % reading input file(s)
[464]202            [Data{iview},tild,errormsg] = read_field(filecell{iview,index},FileType{iview},InputFields{iview},frame_index{iview}(index));
[447]203            if ~isempty(errormsg)
204                errormsg=['error of input reading: ' errormsg];
205                break
[255]206            end
[447]207            if ~isempty(NbSlice_calib)
[454]208                Data{iview}.ZIndex=mod(i1_series{iview}(index)-1,NbSlice_calib{iview})+1;%Zindex for phys transform
[447]209            end
210        end
211        Field=[]; % initiate the current input field structure
212        %%%%%%%%%%%%%%%% end loop on views (input lines) %%%%%%%%%%%%%%%%
213        %%%%%%%%%%%% END STANDARD PART  %%%%%%%%%%%%
214        % EDIT FROM HERE
[462]215       
216        if isempty(errormsg)
[255]217            % coordinate transform (or other user defined transform)
218            if ~isempty(transform_fct)
[27]219                if nbview==2
[41]220                    [Data{1},Data{2}]=transform_fct(Data{1},XmlData{1},Data{2},XmlData{2});
[27]221                    if isempty(Data{2})
222                        Data(2)=[];
223                    end
224                else
[80]225                    Data{1}=transform_fct(Data{1},XmlData{1});
[27]226                end
[255]227            end
228           
[494]229            %% check whether tps is needed, then calculate tps coefficients if needed
230            check_tps=0;
231            if ischar(Param.InputFields.FieldName)
232                Param.InputFields.FieldName={Param.InputFields.FieldName};
233            end
234            for ilist=1:numel(Param.InputFields.FieldName)
235                switch Param.InputFields.FieldName{ilist}
236                    case {'vort','div','strain'}
237                        check_tps=1;
238                end
239            end
240            if strcmp(Param.ProjObject.ProjMode,'filter')
241                check_tps=1;
242            end
243            if check_tps
244                SubDomain=1500; %default, estimated nbre of vectors in a subdomain used for tps
245                if isfield(Data{iview},'SubDomain')
246                    SubDomain=Data{iview}.SubDomain;%
247                end
248                [Data{iview}.SubRange,Data{iview}.NbSites,Data{iview}.Coord_tps,Data{iview}.U_tps,Data{iview}.V_tps,tild,U_smooth,V_smooth,W_smooth,FF] =...
249                    filter_tps([Data{iview}.X(Data{iview}.FF==0) Data{iview}.Y(Data{iview}.FF==0)],Data{iview}.U(Data{iview}.FF==0),Data{iview}.V(Data{iview}.FF==0),[],SubDomain,0);
250                nbvar=numel(Data{iview}.ListVarName);
251                Data{iview}.ListVarName=[Data{iview}.ListVarName {'SubRange','NbSites','Coord_tps','U_tps','V_tps'}];
252                Data{iview}.VarDimName=[Data{iview}.VarDimName {{'nb_coord','nb_bounds','nb_subdomain'},{'nb_subdomain'},...
253                    {'nb_tps','nb_coord','nb_subdomain'},{'nb_tps','nb_subdomain'},{'nb_tps','nb_subdomain'}}];
254                Data{iview}.VarAttribute{nbvar+3}.Role='coord_tps';
255                Data{iview}.VarAttribute{nbvar+4}.Role='vector_x';
256                Data{iview}.VarAttribute{nbvar+5}.Role='vector_y';
257                if isfield(Data{iview},'ListDimName')%cleaning
258                    Data{iview}=rmfield(Data{iview},'ListDimName');
259                end
260                if isfield(Data{iview},'DimValue')%cleaning
261                    Data{iview}=rmfield(Data{iview},'DimValue');
262                end
263            end
264                 
265            % field calculation (vort, div...)   
[478]266            if strcmp(FileType{1},'civx')||strcmp(FileType{1},'civdata')
[494]267                if isfield(Data{1},'Coord_tps')
268                    Data{1}.FieldList=Param.InputFields.FieldName;
269                else
270                    Data{1}=calc_field(Param.InputFields.FieldName,Data{1});%calculate field (vort..)
271                end
[27]272            end
[494]273         
[255]274            % field substration (for two input file series)
[27]275            if length(Data)==2
[494]276                if strcmp(FileType{2},'civx')||strcmp(FileType{2},'civdata')
277                    if isfield(Data{2},'Coord_tps')
278                        Data{2}.FieldList=Param.InputFields.FieldName;
279                    else
280                        Data{2}=calc_field(Param.InputFields.FieldName,Data{2});%calculate field (vort..)
281                    end
[447]282                end
[27]283                [Field,errormsg]=sub_field(Data{1},Data{2}); %substract the two fields
284                if ~isempty(errormsg)
285                    msgbox_uvmat('ERROR',['error in aver_stat/sub_field:' errormsg])
286                    return
287                end
288            else
289                Field=Data{1};
290            end
[451]291           
292            %field projection on an object
[447]293            if Param.CheckObject
[462]294                [Field,errormsg]=proj_field(Field,Param.ProjObject);
[255]295                if ~isempty(errormsg)
[27]296                    msgbox_uvmat('ERROR',['error in aver_stat/proj_field:' errormsg])
297                    return
298                end
[255]299            end
300            nbfiles=nbfiles+1;
[447]301           
302            %%%%%%%%%%%% MAIN RUNNING OPERATIONS  %%%%%%%%%%%%
303            %update sum
[462]304            if nbfiles==1 %first field
305                time_1=[];
306                if isfield(Field,'Time')
307                    time_1=Field.Time(1);
[255]308                end
[462]309                DataOut=Field;%default
310                for ivar=1:length(Field.ListVarName)
311                    VarName=Field.ListVarName{ivar};
312                    DataOut.(VarName)=double(DataOut.(VarName));
313                end
314            else   %current field
315                for ivar=1:length(Field.ListVarName)
316                    VarName=Field.ListVarName{ivar};
317                    sizmean=size(DataOut.(VarName));
318                    siz=size(Field.(VarName));
319                    if ~isequal(DataOut.(VarName),0)&& ~isequal(siz,sizmean)
320                        msgbox_uvmat('ERROR',['unequal size of input field ' VarName ', need to project  on a grid'])
321                        return
322                    else
323                        DataOut.(VarName)=DataOut.(VarName)+ double(Field.(VarName)); % update the sum
324                    end
325                end
[255]326            end
[447]327            %%%%%%%%%%%%   END MAIN RUNNING OPERATIONS  %%%%%%%%%%%%
328        else
[462]329            display(errormsg)
[447]330        end
[255]331    end
[447]332    %%%%%%%%%%%%%%%% end loop on field indices %%%%%%%%%%%%%%%%
333   
[27]334    for ivar=1:length(Field.ListVarName)
335        VarName=Field.ListVarName{ivar};
[447]336        DataOut.(VarName)=DataOut.(VarName)/nbfiles; % normalize the mean
[27]337    end
338    if nbmissing~=0
339        msgbox_uvmat('WARNING',[num2str(nbmissing) ' input files are missing or skipted'])
340    end
[462]341    if isempty(time) % time is read from files
[27]342        if isfield(Field,'Time')
343            time_end=Field.Time(1);%last time read
344            if ~isempty(time_1)
[447]345                DataOut.Time=time_1;
346                DataOut.Time_end=time_end;
[27]347            end
348        end
[457]349    else  % time from ImaDoc prevails if it exists
350        j1=1;%default
351        if ~isempty(j1_series{1})
352            j1=j1_series{1};
353        end
354        DataOut.Time=time(1,i1_series{1}(1),j1filexml);
[454]355        DataOut.Time_end=time(end,i1_series{end}(end),j1_series{end}(end));
[27]356    end
[255]357   
[27]358    %writing the result file
[474]359    OutputFile=fullfile_uvmat(RootPath{1},OutputDir,RootFile{1},FileExtOut,NomTypeOut,i1_series{1}(1),i1_series{1}(end),i_slice,[]);
[447]360    if CheckImage{1} %case of images
361        if isequal(FileInfo{1}.BitDepth,16)||(numel(FileInfo)==2 &&isequal(FileInfo{2}.BitDepth,16))
362            DataOut.A=uint16(DataOut.A);
363            imwrite(DataOut.A,OutputFile,'BitDepth',16); % case of 16 bit images
[27]364        else
[447]365            DataOut.A=uint8(DataOut.A);
366            imwrite(DataOut.A,OutputFile,'BitDepth',8); % case of 16 bit images
[27]367        end
[447]368        display([OutputFile ' written']);
[55]369    else %case of netcdf input file , determine global attributes
[447]370        errormsg=struct2nc(OutputFile,DataOut); %save result file
[27]371        if isempty(errormsg)
[447]372            display([OutputFile ' written']);
[27]373        else
374            msgbox_uvmat('ERROR',['error in writting result file: ' errormsg])
375            display(errormsg)
376        end
[255]377    end  % end averaging  loop
[447]378end
379%%%%%%%%%%%%%%%% end loop on slices %%%%%%%%%%%%%%%%
[255]380
[451]381%% open the result file with uvmat (in RUN mode)
382if checkrun
383    hget_field=findobj(allchild(0),'name','get_field');%find the get_field... GUI
384    delete(hget_field)
385    uvmat(OutputFile)% open the last result file with uvmat
386end
Note: See TracBrowser for help on using the repository browser.